From 52ccb76ab2aca717e89f15b631364904b03299ee Mon Sep 17 00:00:00 2001 From: fingolfin Date: Mon, 3 May 2021 22:10:45 +0200 Subject: [PATCH 01/18] A fix for a warning and a fix for non-standard macro usage. --- yojimbo.h | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/yojimbo.h b/yojimbo.h index 50db5258..e27587bb 100644 --- a/yojimbo.h +++ b/yojimbo.h @@ -3888,7 +3888,7 @@ namespace yojimbo template bool Serialize( Stream & stream ) { (void) stream; return true; } - YOJIMBO_VIRTUAL_SERIALIZE_FUNCTIONS(); + YOJIMBO_VIRTUAL_SERIALIZE_FUNCTIONS() protected: @@ -4152,7 +4152,7 @@ namespace yojimbo #define YOJIMBO_DECLARE_MESSAGE_TYPE( message_type, message_class ) \ \ case message_type: \ - message = YOJIMBO_NEW( allocator, message_class ); \ + message = YOJIMBO_NEW( allocator, message_class, ); \ if ( !message ) \ return NULL; \ SetMessageType( message, message_type ); \ From ae7d547f8defb7c6b8685eb96e91c0e8703a21bc Mon Sep 17 00:00:00 2001 From: fingolfin Date: Tue, 4 May 2021 10:39:12 +0200 Subject: [PATCH 02/18] More fixed warnings. --- shared.h | 8 ++++---- yojimbo.cpp | 4 ++-- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/shared.h b/shared.h index a5496490..f57586ae 100644 --- a/shared.h +++ b/shared.h @@ -72,7 +72,7 @@ struct TestMessage : public Message return true; } - YOJIMBO_VIRTUAL_SERIALIZE_FUNCTIONS(); + YOJIMBO_VIRTUAL_SERIALIZE_FUNCTIONS() }; struct TestBlockMessage : public BlockMessage @@ -90,7 +90,7 @@ struct TestBlockMessage : public BlockMessage return true; } - YOJIMBO_VIRTUAL_SERIALIZE_FUNCTIONS(); + YOJIMBO_VIRTUAL_SERIALIZE_FUNCTIONS() }; struct TestSerializeFailOnReadMessage : public Message @@ -100,7 +100,7 @@ struct TestSerializeFailOnReadMessage : public Message return !Stream::IsReading; } - YOJIMBO_VIRTUAL_SERIALIZE_FUNCTIONS(); + YOJIMBO_VIRTUAL_SERIALIZE_FUNCTIONS() }; struct TestExhaustStreamAllocatorOnReadMessage : public Message @@ -129,7 +129,7 @@ struct TestExhaustStreamAllocatorOnReadMessage : public Message return true; } - YOJIMBO_VIRTUAL_SERIALIZE_FUNCTIONS(); + YOJIMBO_VIRTUAL_SERIALIZE_FUNCTIONS() }; enum TestMessageType diff --git a/yojimbo.cpp b/yojimbo.cpp index 3a244c89..999c0931 100644 --- a/yojimbo.cpp +++ b/yojimbo.cpp @@ -645,7 +645,7 @@ namespace yojimbo return false; if ( m_port != other.m_port ) return false; - if ( m_type == ADDRESS_IPV4 && m_address.ipv4 == other.m_address.ipv4 ) + if ( m_type == ADDRESS_IPV4 && +m_address.ipv4 == +other.m_address.ipv4 ) return true; else if ( m_type == ADDRESS_IPV6 && memcmp( m_address.ipv6, other.m_address.ipv6, sizeof( m_address.ipv6 ) ) == 0 ) return true; @@ -878,7 +878,7 @@ namespace yojimbo m_allocator = &allocator; m_initialized = false; m_matchStatus = MATCH_IDLE; - m_internal = YOJIMBO_NEW( allocator, MatcherInternal ); + m_internal = YOJIMBO_NEW( allocator, MatcherInternal, ); memset( m_connectToken, 0, sizeof( m_connectToken ) ); #else // #if YOJIMBO_WITH_MBEDTLS (void) allocator; From b42ece5f92766a4ee956092bda0cad6119f1d9b8 Mon Sep 17 00:00:00 2001 From: Mike Danylchuk Date: Tue, 28 Sep 2021 09:40:44 -0700 Subject: [PATCH 03/18] Fix assert from delay between reliable messages If there are over 32768 unreliable packets sent between reliable messages, which can happen in under ten minutes at 60 fps, then the sequence number of the next reliable message appears to be old, causing an assertion failure. To avoid this, always interpret outbound sequence values as newer than prior values. Fixes #138 --- test.cpp | 139 ++++++++++++++++++++++++++++++++++++++++++++++++++++ yojimbo.cpp | 4 +- yojimbo.h | 5 +- 3 files changed, 144 insertions(+), 4 deletions(-) diff --git a/test.cpp b/test.cpp index 1703d9db..70c84253 100644 --- a/test.cpp +++ b/test.cpp @@ -802,6 +802,14 @@ void test_sequence_buffer() index--; } + for ( int i = 0; i <= Size; ++i ) + { + TestSequenceData * entry = sequence_buffer.Insert( i, true ); + check( entry ); + entry->sequence = i; + check( sequence_buffer.GetSequence() == i + 1 ); + } + sequence_buffer.Reset(); check( sequence_buffer.GetSequence() == 0 ); @@ -2252,6 +2260,136 @@ void test_reliable_fragment_overflow_bug() server.Stop(); } +void test_reliable_outbound_sequence_outdated() +{ + const uint64_t clientId = 1; + + Address clientAddress( "0.0.0.0", ClientPort ); + Address serverAddress( "127.0.0.1", ServerPort ); + + double time = 100.0; + double deltaTime = 1.0 / 60.0; + + ClientServerConfig config; + config.numChannels = 2; + config.timeout = -1; + + const int BlockSize = config.channel[0].blockFragmentSize * 2; + + Client client( GetDefaultAllocator(), clientAddress, config, adapter, time ); + + uint8_t privateKey[KeyBytes]; + memset( privateKey, 0, KeyBytes ); + + Server server( GetDefaultAllocator(), privateKey, serverAddress, config, adapter, time ); + + server.Start( MaxClients ); + + client.InsecureConnect( privateKey, clientId, serverAddress ); + + Client * clients[] = { &client }; + Server * servers[] = { &server }; + + const int NumIterations = 50000; + + for ( int i = 0; i < NumIterations; ++i ) + { + PumpClientServerUpdate( time, clients, 1, servers, 1 ); + + if ( client.ConnectionFailed() ) + break; + + if ( !client.IsConnecting() && client.IsConnected() && server.GetNumConnectedClients() == 1 ) + break; + } + + check( !client.IsConnecting() ); + check( client.IsConnected() ); + check( server.GetNumConnectedClients() == 1 ); + check( client.GetClientIndex() == 0 ); + check( server.IsClientConnected(0) ); + + int numMessagesSent = 0; + + TestMessage * clientMessage = (TestMessage*) client.CreateMessage( TEST_MESSAGE ); + check( clientMessage ); + client.SendMessage( 0, clientMessage ); + ++numMessagesSent; + + TestBlockMessage * clientBlockMessage = (TestBlockMessage*) client.CreateMessage( TEST_BLOCK_MESSAGE ); + check( clientBlockMessage ); + uint8_t * clientBlockData = client.AllocateBlock( BlockSize ); + memset( clientBlockData, 0, BlockSize ); + client.AttachBlockToMessage( clientBlockMessage, clientBlockData, BlockSize ); + client.SendMessage( 1, clientBlockMessage ); + ++numMessagesSent; + + // Simulate packet sequence being incremented by unreliable messages until it appears outdated. + for ( int i = 0; i < 32000; ++i ) // Test takes much longer when sending 32768 at once. + { + client.SendPackets(); + } + PumpClientServerUpdate( time, clients, 1, servers, 1, deltaTime ); + for ( int j = 0; j < 768; ++j ) + { + client.SendPackets(); + } + + TestMessage * clientMessage2 = (TestMessage*) client.CreateMessage( TEST_MESSAGE ); + check( clientMessage2 ); + client.SendMessage( 0, clientMessage2 ); + ++numMessagesSent; + + TestBlockMessage * clientBlockMessage2 = (TestBlockMessage*) client.CreateMessage( TEST_BLOCK_MESSAGE ); + check( clientBlockMessage2 ); + uint8_t * clientBlockData2 = client.AllocateBlock( BlockSize ); + memset( clientBlockData2, 0, BlockSize ); + client.AttachBlockToMessage( clientBlockMessage2, clientBlockData2, BlockSize ); + client.SendMessage( 1, clientBlockMessage2 ); + ++numMessagesSent; + + int numMessagesReceived = 0; + + for ( int i = 0; i < NumIterations; ++i ) + { + if ( !client.IsConnected() ) + break; + + PumpClientServerUpdate( time, clients, 1, servers, 1, deltaTime ); + + for ( int channelIndex = 0; channelIndex < config.numChannels; ++channelIndex ) + { + Message * messageFromClient = server.ReceiveMessage( 0, channelIndex ); + if ( messageFromClient ) + { + server.ReleaseMessage( 0, messageFromClient ); + ++numMessagesReceived; + } + } + + if ( numMessagesReceived == numMessagesSent ) + break; + } + + check( client.IsConnected() ); + check( server.IsClientConnected( client.GetClientIndex() ) ); + check( numMessagesReceived == numMessagesSent ); + + client.Disconnect(); + + for ( int i = 0; i < NumIterations; ++i ) + { + PumpClientServerUpdate( time, clients, 1, servers, 1 ); + + if ( !client.IsConnected() && server.GetNumConnectedClients() == 0 ) + break; + } + + check( !client.IsConnected() && server.GetNumConnectedClients() == 0 ); + + server.Stop(); +} + void test_single_message_type_reliable() { SingleTestMessageFactory messageFactory( GetDefaultAllocator() ); @@ -2552,6 +2690,7 @@ int main() RUN_TEST( test_client_server_message_exhaust_stream_allocator ); RUN_TEST( test_client_server_message_receive_queue_overflow ); RUN_TEST( test_reliable_fragment_overflow_bug ); + RUN_TEST( test_reliable_outbound_sequence_outdated ); RUN_TEST( test_single_message_type_reliable ); RUN_TEST( test_single_message_type_reliable_blocks ); RUN_TEST( test_single_message_type_unreliable ); diff --git a/yojimbo.cpp b/yojimbo.cpp index 2298021b..3eb247ff 100644 --- a/yojimbo.cpp +++ b/yojimbo.cpp @@ -1885,7 +1885,7 @@ namespace yojimbo void ReliableOrderedChannel::AddMessagePacketEntry( const uint16_t * messageIds, int numMessageIds, uint16_t sequence ) { - SentPacketEntry * sentPacket = m_sentPackets->Insert( sequence ); + SentPacketEntry * sentPacket = m_sentPackets->Insert( sequence, true ); yojimbo_assert( sentPacket ); if ( sentPacket ) { @@ -2167,7 +2167,7 @@ namespace yojimbo void ReliableOrderedChannel::AddFragmentPacketEntry( uint16_t messageId, uint16_t fragmentId, uint16_t sequence ) { - SentPacketEntry * sentPacket = m_sentPackets->Insert( sequence ); + SentPacketEntry * sentPacket = m_sentPackets->Insert( sequence, true ); yojimbo_assert( sentPacket ); if ( sentPacket ) { diff --git a/yojimbo.h b/yojimbo.h index a10bd8b6..11d4effd 100644 --- a/yojimbo.h +++ b/yojimbo.h @@ -1412,12 +1412,13 @@ namespace yojimbo Insert an entry in the sequence buffer. IMPORTANT: If another entry exists at the sequence modulo buffer size, it is overwritten. @param sequence The sequence number. + @param guaranteed_order Whether sequence is always the newest value (when sending) or can be out of order (when receiving). @returns The sequence buffer entry, which you must fill with your data. NULL if a sequence buffer entry could not be added for your sequence number (if the sequence number is too old for example). */ - T * Insert( uint16_t sequence ) + T * Insert( uint16_t sequence, bool guaranteed_order = false ) { - if ( sequence_greater_than( sequence + 1, m_sequence ) ) + if ( sequence_greater_than( sequence + 1, m_sequence ) || guaranteed_order ) { RemoveEntries( m_sequence, sequence ); m_sequence = sequence + 1; From 456882bdc1cf2a6d6642cb6711643282b7eff4cb Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?J=C3=B8rgen=20P=2E=20Tjern=C3=B8?= Date: Wed, 9 Mar 2022 12:03:34 -0800 Subject: [PATCH 04/18] Leak detection: Use registered printf & assert Instead of directly calling `printf` and `exit`, use `yojimbo_printf` and `yojimbo_assert` in the leak detection in `Allocator::~Allocator`. This helps integrate better with client applications. Also switch the `default_assert_handler` to use `yojimbo_printf`. --- yojimbo.cpp | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/yojimbo.cpp b/yojimbo.cpp index ccd6a961..09e31211 100644 --- a/yojimbo.cpp +++ b/yojimbo.cpp @@ -195,16 +195,16 @@ namespace yojimbo #if YOJIMBO_DEBUG_MEMORY_LEAKS if ( m_alloc_map.size() ) { - printf( "you leaked memory!\n\n" ); + yojimbo_printf( YOJIMBO_LOG_LEVEL_ERROR, "you leaked memory!\n\n" ); typedef std::map::iterator itor_type; for ( itor_type i = m_alloc_map.begin(); i != m_alloc_map.end(); ++i ) { void * p = i->first; AllocatorEntry entry = i->second; - printf( "leaked block %p (%d bytes) - %s:%d\n", p, (int) entry.size, entry.file, entry.line ); + yojimbo_printf( YOJIMBO_LOG_LEVEL_ERROR, "leaked block %p (%d bytes) - %s:%d\n", p, (int) entry.size, entry.file, entry.line ); } - printf( "\n" ); - exit(1); + yojimbo_printf( YOJIMBO_LOG_LEVEL_ERROR, "\n" ); + yojimbo_assert( false && "Leaks detected, see log" ); } #endif // #if YOJIMBO_DEBUG_MEMORY_LEAKS } @@ -669,7 +669,7 @@ namespace yojimbo static void default_assert_handler( const char * condition, const char * function, const char * file, int line ) { - printf( "assert failed: ( %s ), function %s, file %s, line %d\n", condition, function, file, line ); + yojimbo_printf( YOJIMBO_LOG_LEVEL_NONE, "assert failed: ( %s ), function %s, file %s, line %d\n", condition, function, file, line ); #if defined( __GNUC__ ) __builtin_trap(); #elif defined( _MSC_VER ) From ed5b6abe7f1bdf04ae38381082bcba4c37f7114c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?J=C3=B8rgen=20P=2E=20Tjern=C3=B8?= Date: Wed, 21 Dec 2022 11:03:14 -0800 Subject: [PATCH 05/18] Add a comment about YOJIMBO_LOG_LEVEL_NONE --- yojimbo.cpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/yojimbo.cpp b/yojimbo.cpp index 09e31211..12f89870 100644 --- a/yojimbo.cpp +++ b/yojimbo.cpp @@ -669,6 +669,8 @@ namespace yojimbo static void default_assert_handler( const char * condition, const char * function, const char * file, int line ) { + // We use YOJIMBO_LOG_LEVEL_NONE because it's lower than YOJIMBO_LOG_LEVEL_ERROR, so even if you suppress errors (by setting + // yojimbo_log_level(YOJIMBO_LOG_LEVEL_NONE)), this will still be logged. yojimbo_printf( YOJIMBO_LOG_LEVEL_NONE, "assert failed: ( %s ), function %s, file %s, line %d\n", condition, function, file, line ); #if defined( __GNUC__ ) __builtin_trap(); From cdde75c79fd7bc0f139d1413a4fd1b380b4e712b Mon Sep 17 00:00:00 2001 From: kbirk Date: Sun, 19 Mar 2023 19:57:53 -0400 Subject: [PATCH 06/18] Keep state as connecting when netcode state is NETCODE_CLIENT_STATE_SENDING_CONNECTION_RESPONSE --- yojimbo.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/yojimbo.cpp b/yojimbo.cpp index b0d32e2b..53c2878e 100644 --- a/yojimbo.cpp +++ b/yojimbo.cpp @@ -3301,7 +3301,7 @@ namespace yojimbo Disconnect(); SetClientState( CLIENT_STATE_DISCONNECTED ); } - else if ( state == NETCODE_CLIENT_STATE_SENDING_CONNECTION_REQUEST ) + else if ( state == NETCODE_CLIENT_STATE_SENDING_CONNECTION_REQUEST || state == NETCODE_CLIENT_STATE_SENDING_CONNECTION_RESPONSE ) { SetClientState( CLIENT_STATE_CONNECTING ); } From 3fae585f5d43203f81956b92e17ed68e75dae7ca Mon Sep 17 00:00:00 2001 From: Glenn Fiedler Date: Fri, 22 Dec 2023 17:13:41 -0500 Subject: [PATCH 07/18] Create FUNDING.yml --- .github/FUNDING.yml | 3 +++ 1 file changed, 3 insertions(+) create mode 100644 .github/FUNDING.yml diff --git a/.github/FUNDING.yml b/.github/FUNDING.yml new file mode 100644 index 00000000..3515148e --- /dev/null +++ b/.github/FUNDING.yml @@ -0,0 +1,3 @@ +# These are supported funding model platforms + +github: mas-bandwidth From e134d7959482875975e388d14c064205070c8ca1 Mon Sep 17 00:00:00 2001 From: Glenn Fiedler Date: Sat, 23 Dec 2023 19:19:16 -0500 Subject: [PATCH 08/18] wip cleaning up yojimbo --- certs.c => certs/certs.c | 0 certs.h => certs/certs.h | 0 netcode.io | 1 - netcode/netcode.c | 8963 +++++++++++++++++ netcode/netcode.h | 293 + premake5.lua | 428 +- reliable.io | 1 - reliable/reliable.c | 2418 +++++ reliable/reliable.h | 176 + sodium/sodium.h | 48 + sodium/sodium_aead_chacha20poly1305.c | 400 + sodium/sodium_aead_xchacha20poly1305.c | 262 + sodium/sodium_blake2.h | 102 + sodium/sodium_blake2b-compress-avx2.c | 51 + sodium/sodium_blake2b-compress-avx2.h | 142 + sodium/sodium_blake2b-compress-ref.c | 93 + sodium/sodium_blake2b-compress-sse41.c | 89 + sodium/sodium_blake2b-compress-sse41.h | 103 + sodium/sodium_blake2b-compress-ssse3.c | 92 + sodium/sodium_blake2b-compress-ssse3.h | 103 + sodium/sodium_blake2b-load-avx2.h | 340 + sodium/sodium_blake2b-load-sse2.h | 164 + sodium/sodium_blake2b-load-sse41.h | 307 + sodium/sodium_blake2b-ref.c | 453 + .../sodium_box_curve25519xchacha20poly1305.c | 204 + .../sodium_box_curve25519xsalsa20poly1305.c | 156 + ...ium_box_seal_curve25519xchacha20poly1305.c | 79 + sodium/sodium_chacha20-ref.c | 312 + sodium/sodium_chacha20-ref.h | 8 + sodium/sodium_codecs.c | 333 + sodium/sodium_core.c | 218 + windows/sodium/core.h => sodium/sodium_core.h | 10 +- sodium/sodium_core_ed25519.c | 195 + sodium/sodium_core_hchacha20.c | 93 + sodium/sodium_core_hsalsa20.c | 21 + sodium/sodium_core_hsalsa20_ref2.c | 95 + sodium/sodium_core_salsa_ref.c | 195 + .../sodium_crypto_aead_chacha20poly1305.h | 65 +- .../sodium_crypto_aead_xchacha20poly1305.h | 38 +- .../sodium_crypto_auth.h | 18 +- .../sodium_crypto_auth_hmacsha512.h | 27 +- .../sodium_crypto_auth_hmacsha512256.h | 25 +- sodium/sodium_crypto_box.c | 47 + .../sodium_crypto_box.h | 72 +- ...m_crypto_box_curve25519xchacha20poly1305.h | 58 +- ...um_crypto_box_curve25519xsalsa20poly1305.h | 60 +- sodium/sodium_crypto_box_easy.c | 115 + sodium/sodium_crypto_core_ed25519.h | 68 + .../sodium_crypto_core_hchacha20.h | 10 +- .../sodium_crypto_core_hsalsa20.h | 10 +- .../sodium_crypto_core_salsa20.h | 10 +- .../sodium_crypto_core_salsa2012.h | 10 +- .../sodium_crypto_core_salsa208.h | 22 +- sodium/sodium_crypto_generichash.c | 91 + .../sodium_crypto_generichash.h | 36 +- .../sodium_crypto_generichash_blake2b.h | 49 +- .../sodium_crypto_hash.h | 9 +- .../sodium_crypto_hash_sha512.h | 19 +- sodium/sodium_crypto_kx.c | 103 + .../crypto_kx.h => sodium/sodium_crypto_kx.h | 21 +- .../sodium_crypto_onetimeauth.h | 29 +- .../sodium_crypto_onetimeauth_poly1305.h | 28 +- sodium/sodium_crypto_scalarmult.c | 33 + .../sodium_crypto_scalarmult.h | 24 +- .../sodium_crypto_scalarmult_curve25519.h | 20 +- sodium/sodium_crypto_scalarmult_ed25519.h | 45 + sodium/sodium_crypto_secretbox.c | 25 + .../sodium_crypto_secretbox.h | 36 +- sodium/sodium_crypto_secretbox_easy.c | 144 + ...odium_crypto_secretbox_xchacha20poly1305.h | 24 +- ...sodium_crypto_secretbox_xsalsa20poly1305.h | 40 +- ...um_crypto_secretstream_xchacha20poly1305.h | 93 + .../sodium_crypto_shorthash.h | 15 +- .../sodium_crypto_shorthash_siphash24.h | 14 +- sodium/sodium_crypto_sign.c | 51 + .../sodium_crypto_sign.h | 47 +- .../sodium_crypto_sign_ed25519.h | 62 +- ...dium_crypto_sign_edwards25519sha512batch.h | 12 +- .../sodium_crypto_stream.h | 22 +- .../sodium_crypto_stream_chacha20.h | 46 +- .../sodium_crypto_stream_salsa20.h | 23 +- .../sodium_crypto_stream_salsa2012.h | 19 +- .../sodium_crypto_stream_salsa208.h | 26 +- .../sodium_crypto_stream_xchacha20.h | 23 +- .../sodium_crypto_stream_xsalsa20.h | 23 +- .../sodium_crypto_verify_16.h | 7 +- .../sodium_crypto_verify_32.h | 6 +- .../sodium_crypto_verify_64.h | 6 +- sodium/sodium_dolbeau_chacha20-avx2.c | 179 + sodium/sodium_dolbeau_chacha20-avx2.h | 8 + sodium/sodium_dolbeau_chacha20-ssse3.c | 173 + sodium/sodium_dolbeau_chacha20-ssse3.h | 8 + sodium/sodium_dolbeau_u0.h | 86 + sodium/sodium_dolbeau_u1.h | 98 + sodium/sodium_dolbeau_u4.h | 177 + sodium/sodium_dolbeau_u8.h | 357 + sodium/sodium_export.h | 27 + sodium/sodium_fe_25_5_base.h | 1344 +++ sodium/sodium_fe_25_5_base2.h | 40 + sodium/sodium_fe_25_5_constants.h | 20 + sodium/sodium_fe_25_5_fe.h | 220 + sodium/sodium_fe_51_base.h | 1344 +++ sodium/sodium_fe_51_base2.h | 40 + sodium/sodium_fe_51_constants.h | 21 + sodium/sodium_fe_51_fe.h | 116 + sodium/sodium_generichash_blake2.c | 55 + sodium/sodium_generichash_blake2b.c | 117 + sodium/sodium_hash_sha512.c | 13 + sodium/sodium_hash_sha512_cp.c | 282 + sodium/sodium_onetimeauth_poly1305.c | 97 + sodium/sodium_onetimeauth_poly1305.h | 21 + sodium/sodium_poly1305-sse2.c | 951 ++ sodium/sodium_poly1305-sse2.h | 12 + sodium/sodium_poly1305_donna.c | 124 + sodium/sodium_poly1305_donna.h | 12 + sodium/sodium_poly1305_donna32.h | 235 + sodium/sodium_poly1305_donna64.h | 220 + sodium/sodium_private_chacha20_ietf_ext.h | 16 + sodium/sodium_private_common.h | 317 + sodium/sodium_private_ed25519_ref10.h | 127 + sodium/sodium_private_ed25519_ref10_fe_25_5.h | 1060 ++ sodium/sodium_private_ed25519_ref10_fe_51.h | 518 + sodium/sodium_private_implementations.h | 10 + sodium/sodium_private_mutex.h | 7 + sodium/sodium_private_sse2_64_32.h | 50 + sodium/sodium_randombytes.c | 207 + .../sodium_randombytes.h | 26 +- sodium/sodium_randombytes_nativeclient.h | 22 + sodium/sodium_randombytes_salsa20_random.c | 568 ++ .../sodium_randombytes_salsa20_random.h | 5 +- sodium/sodium_randombytes_sysrandom.c | 422 + .../sodium_randombytes_sysrandom.h | 5 +- sodium/sodium_ref10_ed25519.c | 2601 +++++ sodium/sodium_ref10_keypair.c | 83 + sodium/sodium_ref10_open.c | 96 + .../sodium_ref10_scalarmult_ed25519_ref10.c | 118 + sodium/sodium_ref10_sign.c | 143 + sodium/sodium_ref10_sign_ed25519.h | 18 + sodium/sodium_ref10_x25519.c | 180 + sodium/sodium_ref10_x25519.h | 10 + sodium/sodium_runtime.c | 319 + .../runtime.h => sodium/sodium_runtime.h | 15 +- sodium/sodium_salsa20-ref.c | 124 + sodium/sodium_salsa20-ref.h | 8 + sodium/sodium_salsa20-xmm6.c | 33 + sodium/sodium_salsa20-xmm6.h | 8 + sodium/sodium_salsa20-xmm6_asm.S | 962 ++ sodium/sodium_sandy2x.S | 19 + sodium/sodium_sandy2x_consts.S | 27 + sodium/sodium_sandy2x_consts_namespace.h | 20 + sodium/sodium_sandy2x_curve25519.c | 118 + sodium/sodium_sandy2x_curve25519.h | 9 + sodium/sodium_sandy2x_fe.h | 26 + sodium/sodium_sandy2x_fe51.h | 35 + sodium/sodium_sandy2x_fe51_invert.c | 61 + sodium/sodium_sandy2x_fe51_mul.S | 199 + sodium/sodium_sandy2x_fe51_namespace.h | 16 + sodium/sodium_sandy2x_fe51_nsquare.S | 174 + sodium/sodium_sandy2x_fe51_pack.S | 228 + sodium/sodium_sandy2x_fe_frombytes.c | 81 + sodium/sodium_sandy2x_ladder.S | 1442 +++ sodium/sodium_sandy2x_ladder.h | 18 + sodium/sodium_sandy2x_ladder_base.S | 1297 +++ sodium/sodium_sandy2x_ladder_base.h | 18 + sodium/sodium_sandy2x_ladder_base_namespace.h | 8 + sodium/sodium_sandy2x_ladder_namespace.h | 8 + sodium/sodium_scalarmult_curve25519.c | 68 + sodium/sodium_scalarmult_curve25519.h | 11 + sodium/sodium_secretbox_xchacha20poly1305.c | 177 + sodium/sodium_secretbox_xsalsa20poly1305.c | 89 + sodium/sodium_sign_ed25519.c | 97 + sodium/sodium_stream_chacha20.c | 194 + sodium/sodium_stream_chacha20.h | 22 + sodium/sodium_stream_salsa20.c | 114 + sodium/sodium_stream_salsa20.h | 16 + sodium/sodium_stream_xchacha20.c | 69 + sodium/sodium_stream_xsalsa20.c | 66 + sodium/sodium_utils.c | 795 ++ .../sodium/utils.h => sodium/sodium_utils.h | 84 +- sodium/sodium_verify.c | 98 + sodium/sodium_version.c | 30 + .../version.h => sodium/sodium_version.h | 14 +- sodium/sodium_xmm6int_salsa20-avx2.c | 133 + sodium/sodium_xmm6int_salsa20-avx2.h | 8 + sodium/sodium_xmm6int_salsa20-sse2.c | 124 + sodium/sodium_xmm6int_salsa20-sse2.h | 8 + sodium/sodium_xmm6int_u0.h | 195 + sodium/sodium_xmm6int_u1.h | 207 + sodium/sodium_xmm6int_u4.h | 547 + sodium/sodium_xmm6int_u8.h | 477 + windows/mbedtls/aes.h | 626 -- windows/mbedtls/aria.h | 356 - windows/mbedtls/asn1.h | 604 -- windows/mbedtls/asn1write.h | 366 - windows/mbedtls/base64.h | 92 - windows/mbedtls/bignum.h | 1021 -- windows/mbedtls/build_info.h | 83 - windows/mbedtls/camellia.h | 314 - windows/mbedtls/ccm.h | 508 - windows/mbedtls/chacha20.h | 214 - windows/mbedtls/chachapoly.h | 354 - windows/mbedtls/check_config.h | 838 -- windows/mbedtls/cipher.h | 983 -- windows/mbedtls/cmac.h | 223 - windows/mbedtls/compat-2.x.h | 58 - windows/mbedtls/config_psa.h | 770 -- windows/mbedtls/ctr_drbg.h | 575 -- windows/mbedtls/debug.h | 311 - windows/mbedtls/des.h | 348 - windows/mbedtls/dhm.h | 976 -- windows/mbedtls/ecdh.h | 458 - windows/mbedtls/ecdsa.h | 506 - windows/mbedtls/ecjpake.h | 287 - windows/mbedtls/ecp.h | 1286 --- windows/mbedtls/entropy.h | 288 - windows/mbedtls/error.h | 210 - windows/mbedtls/gcm.h | 379 - windows/mbedtls/hkdf.h | 135 - windows/mbedtls/hmac_drbg.h | 443 - windows/mbedtls/mbedtls_config.h | 3263 ------ windows/mbedtls/md.h | 446 - windows/mbedtls/md5.h | 203 - windows/mbedtls/memory_buffer_alloc.h | 145 - windows/mbedtls/net_sockets.h | 285 - windows/mbedtls/nist_kw.h | 179 - windows/mbedtls/oid.h | 641 -- windows/mbedtls/pem.h | 141 - windows/mbedtls/pk.h | 890 -- windows/mbedtls/pkcs12.h | 105 - windows/mbedtls/pkcs5.h | 103 - windows/mbedtls/platform.h | 411 - windows/mbedtls/platform_time.h | 76 - windows/mbedtls/platform_util.h | 122 - windows/mbedtls/poly1305.h | 180 - windows/mbedtls/private_access.h | 32 - windows/mbedtls/psa_util.h | 503 - windows/mbedtls/ripemd160.h | 149 - windows/mbedtls/rsa.h | 1119 -- windows/mbedtls/sha1.h | 231 - windows/mbedtls/sha256.h | 191 - windows/mbedtls/sha512.h | 204 - windows/mbedtls/ssl.h | 4152 -------- windows/mbedtls/ssl_cache.h | 161 - windows/mbedtls/ssl_ciphersuites.h | 528 - windows/mbedtls/ssl_cookie.h | 110 - windows/mbedtls/ssl_ticket.h | 137 - windows/mbedtls/threading.h | 116 - windows/mbedtls/timing.h | 97 - windows/mbedtls/version.h | 90 - windows/mbedtls/x509.h | 347 - windows/mbedtls/x509_crl.h | 171 - windows/mbedtls/x509_crt.h | 1179 --- windows/mbedtls/x509_csr.h | 300 - windows/sodium.h | 68 - windows/sodium.lib | Bin 1920144 -> 0 bytes windows/sodium/crypto_aead_aes256gcm.h | 145 - windows/sodium/crypto_auth_hmacsha256.h | 68 - windows/sodium/crypto_hash_sha256.h | 57 - windows/sodium/crypto_kdf.h | 51 - windows/sodium/crypto_kdf_blake2b.h | 42 - windows/sodium/crypto_pwhash.h | 125 - windows/sodium/crypto_pwhash_argon2i.h | 116 - windows/sodium/crypto_pwhash_argon2id.h | 116 - .../crypto_pwhash_scryptsalsa208sha256.h | 112 - windows/sodium/crypto_stream_aes128ctr.h | 65 - windows/sodium/export.h | 44 - 266 files changed, 39697 insertions(+), 32727 deletions(-) rename certs.c => certs/certs.c (100%) rename certs.h => certs/certs.h (100%) delete mode 160000 netcode.io create mode 100755 netcode/netcode.c create mode 100755 netcode/netcode.h delete mode 160000 reliable.io create mode 100644 reliable/reliable.c create mode 100644 reliable/reliable.h create mode 100644 sodium/sodium.h create mode 100644 sodium/sodium_aead_chacha20poly1305.c create mode 100644 sodium/sodium_aead_xchacha20poly1305.c create mode 100644 sodium/sodium_blake2.h create mode 100644 sodium/sodium_blake2b-compress-avx2.c create mode 100644 sodium/sodium_blake2b-compress-avx2.h create mode 100644 sodium/sodium_blake2b-compress-ref.c create mode 100644 sodium/sodium_blake2b-compress-sse41.c create mode 100644 sodium/sodium_blake2b-compress-sse41.h create mode 100644 sodium/sodium_blake2b-compress-ssse3.c create mode 100644 sodium/sodium_blake2b-compress-ssse3.h create mode 100644 sodium/sodium_blake2b-load-avx2.h create mode 100644 sodium/sodium_blake2b-load-sse2.h create mode 100644 sodium/sodium_blake2b-load-sse41.h create mode 100644 sodium/sodium_blake2b-ref.c create mode 100644 sodium/sodium_box_curve25519xchacha20poly1305.c create mode 100644 sodium/sodium_box_curve25519xsalsa20poly1305.c create mode 100644 sodium/sodium_box_seal_curve25519xchacha20poly1305.c create mode 100644 sodium/sodium_chacha20-ref.c create mode 100644 sodium/sodium_chacha20-ref.h create mode 100644 sodium/sodium_codecs.c create mode 100644 sodium/sodium_core.c rename windows/sodium/core.h => sodium/sodium_core.h (55%) create mode 100644 sodium/sodium_core_ed25519.c create mode 100644 sodium/sodium_core_hchacha20.c create mode 100644 sodium/sodium_core_hsalsa20.c create mode 100644 sodium/sodium_core_hsalsa20_ref2.c create mode 100644 sodium/sodium_core_salsa_ref.c rename windows/sodium/crypto_aead_chacha20poly1305.h => sodium/sodium_crypto_aead_chacha20poly1305.h (79%) rename windows/sodium/crypto_aead_xchacha20poly1305.h => sodium/sodium_crypto_aead_xchacha20poly1305.h (76%) rename windows/sodium/crypto_auth.h => sodium/sodium_crypto_auth.h (72%) rename windows/sodium/crypto_auth_hmacsha512.h => sodium/sodium_crypto_auth_hmacsha512.h (73%) rename windows/sodium/crypto_auth_hmacsha512256.h => sodium/sodium_crypto_auth_hmacsha512256.h (75%) create mode 100644 sodium/sodium_crypto_box.c rename windows/sodium/crypto_box.h => sodium/sodium_crypto_box.h (75%) rename windows/sodium/crypto_box_curve25519xchacha20poly1305.h => sodium/sodium_crypto_box_curve25519xchacha20poly1305.h (83%) rename windows/sodium/crypto_box_curve25519xsalsa20poly1305.h => sodium/sodium_crypto_box_curve25519xsalsa20poly1305.h (79%) create mode 100644 sodium/sodium_crypto_box_easy.c create mode 100644 sodium/sodium_crypto_core_ed25519.h rename windows/sodium/crypto_core_hchacha20.h => sodium/sodium_crypto_core_hchacha20.h (86%) rename windows/sodium/crypto_core_hsalsa20.h => sodium/sodium_crypto_core_hsalsa20.h (86%) rename windows/sodium/crypto_core_salsa20.h => sodium/sodium_crypto_core_salsa20.h (86%) rename windows/sodium/crypto_core_salsa2012.h => sodium/sodium_crypto_core_salsa2012.h (86%) rename windows/sodium/crypto_core_salsa208.h => sodium/sodium_crypto_core_salsa208.h (52%) create mode 100644 sodium/sodium_crypto_generichash.c rename windows/sodium/crypto_generichash.h => sodium/sodium_crypto_generichash.h (76%) rename windows/sodium/crypto_generichash_blake2b.h => sodium/sodium_crypto_generichash_blake2b.h (83%) rename windows/sodium/crypto_hash.h => sodium/sodium_crypto_hash.h (84%) rename windows/sodium/crypto_hash_sha512.h => sodium/sodium_crypto_hash_sha512.h (72%) create mode 100644 sodium/sodium_crypto_kx.c rename windows/sodium/crypto_kx.h => sodium/sodium_crypto_kx.h (86%) rename windows/sodium/crypto_onetimeauth.h => sodium/sodium_crypto_onetimeauth.h (71%) rename windows/sodium/crypto_onetimeauth_poly1305.h => sodium/sodium_crypto_onetimeauth_poly1305.h (75%) create mode 100644 sodium/sodium_crypto_scalarmult.c rename windows/sodium/crypto_scalarmult.h => sodium/sodium_crypto_scalarmult.h (59%) rename windows/sodium/crypto_scalarmult_curve25519.h => sodium/sodium_crypto_scalarmult_curve25519.h (50%) create mode 100644 sodium/sodium_crypto_scalarmult_ed25519.h create mode 100644 sodium/sodium_crypto_secretbox.c rename windows/sodium/crypto_secretbox.h => sodium/sodium_crypto_secretbox.h (75%) create mode 100644 sodium/sodium_crypto_secretbox_easy.c rename windows/sodium/crypto_secretbox_xchacha20poly1305.h => sodium/sodium_crypto_secretbox_xchacha20poly1305.h (79%) rename windows/sodium/crypto_secretbox_xsalsa20poly1305.h => sodium/sodium_crypto_secretbox_xsalsa20poly1305.h (68%) create mode 100644 sodium/sodium_crypto_secretstream_xchacha20poly1305.h rename windows/sodium/crypto_shorthash.h => sodium/sodium_crypto_shorthash.h (81%) rename windows/sodium/crypto_shorthash_siphash24.h => sodium/sodium_crypto_shorthash_siphash24.h (87%) create mode 100644 sodium/sodium_crypto_sign.c rename windows/sodium/crypto_sign.h => sodium/sodium_crypto_sign.h (67%) rename windows/sodium/crypto_sign_ed25519.h => sodium/sodium_crypto_sign_ed25519.h (70%) rename windows/sodium/crypto_sign_edwards25519sha512batch.h => sodium/sodium_crypto_sign_edwards25519sha512batch.h (81%) rename windows/sodium/crypto_stream.h => sodium/sodium_crypto_stream.h (76%) rename windows/sodium/crypto_stream_chacha20.h => sodium/sodium_crypto_stream_chacha20.h (70%) rename windows/sodium/crypto_stream_salsa20.h => sodium/sodium_crypto_stream_salsa20.h (74%) rename windows/sodium/crypto_stream_salsa2012.h => sodium/sodium_crypto_stream_salsa2012.h (75%) rename windows/sodium/crypto_stream_salsa208.h => sodium/sodium_crypto_stream_salsa208.h (60%) rename windows/sodium/crypto_stream_xchacha20.h => sodium/sodium_crypto_stream_xchacha20.h (74%) rename windows/sodium/crypto_stream_xsalsa20.h => sodium/sodium_crypto_stream_xsalsa20.h (74%) rename windows/sodium/crypto_verify_16.h => sodium/sodium_crypto_verify_16.h (73%) rename windows/sodium/crypto_verify_32.h => sodium/sodium_crypto_verify_32.h (74%) rename windows/sodium/crypto_verify_64.h => sodium/sodium_crypto_verify_64.h (74%) create mode 100644 sodium/sodium_dolbeau_chacha20-avx2.c create mode 100644 sodium/sodium_dolbeau_chacha20-avx2.h create mode 100644 sodium/sodium_dolbeau_chacha20-ssse3.c create mode 100644 sodium/sodium_dolbeau_chacha20-ssse3.h create mode 100644 sodium/sodium_dolbeau_u0.h create mode 100644 sodium/sodium_dolbeau_u1.h create mode 100644 sodium/sodium_dolbeau_u4.h create mode 100644 sodium/sodium_dolbeau_u8.h create mode 100644 sodium/sodium_export.h create mode 100644 sodium/sodium_fe_25_5_base.h create mode 100644 sodium/sodium_fe_25_5_base2.h create mode 100644 sodium/sodium_fe_25_5_constants.h create mode 100644 sodium/sodium_fe_25_5_fe.h create mode 100644 sodium/sodium_fe_51_base.h create mode 100644 sodium/sodium_fe_51_base2.h create mode 100644 sodium/sodium_fe_51_constants.h create mode 100644 sodium/sodium_fe_51_fe.h create mode 100644 sodium/sodium_generichash_blake2.c create mode 100644 sodium/sodium_generichash_blake2b.c create mode 100644 sodium/sodium_hash_sha512.c create mode 100644 sodium/sodium_hash_sha512_cp.c create mode 100644 sodium/sodium_onetimeauth_poly1305.c create mode 100644 sodium/sodium_onetimeauth_poly1305.h create mode 100644 sodium/sodium_poly1305-sse2.c create mode 100644 sodium/sodium_poly1305-sse2.h create mode 100644 sodium/sodium_poly1305_donna.c create mode 100644 sodium/sodium_poly1305_donna.h create mode 100644 sodium/sodium_poly1305_donna32.h create mode 100644 sodium/sodium_poly1305_donna64.h create mode 100644 sodium/sodium_private_chacha20_ietf_ext.h create mode 100644 sodium/sodium_private_common.h create mode 100644 sodium/sodium_private_ed25519_ref10.h create mode 100644 sodium/sodium_private_ed25519_ref10_fe_25_5.h create mode 100644 sodium/sodium_private_ed25519_ref10_fe_51.h create mode 100644 sodium/sodium_private_implementations.h create mode 100644 sodium/sodium_private_mutex.h create mode 100644 sodium/sodium_private_sse2_64_32.h create mode 100644 sodium/sodium_randombytes.c rename windows/sodium/randombytes.h => sodium/sodium_randombytes.h (79%) create mode 100644 sodium/sodium_randombytes_nativeclient.h create mode 100644 sodium/sodium_randombytes_salsa20_random.c rename windows/sodium/randombytes_salsa20_random.h => sodium/sodium_randombytes_salsa20_random.h (79%) create mode 100644 sodium/sodium_randombytes_sysrandom.c rename windows/sodium/randombytes_sysrandom.h => sodium/sodium_randombytes_sysrandom.h (79%) create mode 100644 sodium/sodium_ref10_ed25519.c create mode 100644 sodium/sodium_ref10_keypair.c create mode 100644 sodium/sodium_ref10_open.c create mode 100644 sodium/sodium_ref10_scalarmult_ed25519_ref10.c create mode 100644 sodium/sodium_ref10_sign.c create mode 100644 sodium/sodium_ref10_sign_ed25519.h create mode 100644 sodium/sodium_ref10_x25519.c create mode 100644 sodium/sodium_ref10_x25519.h create mode 100644 sodium/sodium_runtime.c rename windows/sodium/runtime.h => sodium/sodium_runtime.h (79%) create mode 100644 sodium/sodium_salsa20-ref.c create mode 100644 sodium/sodium_salsa20-ref.h create mode 100644 sodium/sodium_salsa20-xmm6.c create mode 100644 sodium/sodium_salsa20-xmm6.h create mode 100644 sodium/sodium_salsa20-xmm6_asm.S create mode 100644 sodium/sodium_sandy2x.S create mode 100644 sodium/sodium_sandy2x_consts.S create mode 100644 sodium/sodium_sandy2x_consts_namespace.h create mode 100644 sodium/sodium_sandy2x_curve25519.c create mode 100644 sodium/sodium_sandy2x_curve25519.h create mode 100644 sodium/sodium_sandy2x_fe.h create mode 100644 sodium/sodium_sandy2x_fe51.h create mode 100644 sodium/sodium_sandy2x_fe51_invert.c create mode 100644 sodium/sodium_sandy2x_fe51_mul.S create mode 100644 sodium/sodium_sandy2x_fe51_namespace.h create mode 100644 sodium/sodium_sandy2x_fe51_nsquare.S create mode 100644 sodium/sodium_sandy2x_fe51_pack.S create mode 100644 sodium/sodium_sandy2x_fe_frombytes.c create mode 100644 sodium/sodium_sandy2x_ladder.S create mode 100644 sodium/sodium_sandy2x_ladder.h create mode 100644 sodium/sodium_sandy2x_ladder_base.S create mode 100644 sodium/sodium_sandy2x_ladder_base.h create mode 100644 sodium/sodium_sandy2x_ladder_base_namespace.h create mode 100644 sodium/sodium_sandy2x_ladder_namespace.h create mode 100644 sodium/sodium_scalarmult_curve25519.c create mode 100644 sodium/sodium_scalarmult_curve25519.h create mode 100644 sodium/sodium_secretbox_xchacha20poly1305.c create mode 100644 sodium/sodium_secretbox_xsalsa20poly1305.c create mode 100644 sodium/sodium_sign_ed25519.c create mode 100644 sodium/sodium_stream_chacha20.c create mode 100644 sodium/sodium_stream_chacha20.h create mode 100644 sodium/sodium_stream_salsa20.c create mode 100644 sodium/sodium_stream_salsa20.h create mode 100644 sodium/sodium_stream_xchacha20.c create mode 100644 sodium/sodium_stream_xsalsa20.c create mode 100644 sodium/sodium_utils.c rename windows/sodium/utils.h => sodium/sodium_utils.h (56%) create mode 100644 sodium/sodium_verify.c create mode 100644 sodium/sodium_version.c rename windows/sodium/version.h => sodium/sodium_version.h (61%) create mode 100644 sodium/sodium_xmm6int_salsa20-avx2.c create mode 100644 sodium/sodium_xmm6int_salsa20-avx2.h create mode 100644 sodium/sodium_xmm6int_salsa20-sse2.c create mode 100644 sodium/sodium_xmm6int_salsa20-sse2.h create mode 100644 sodium/sodium_xmm6int_u0.h create mode 100644 sodium/sodium_xmm6int_u1.h create mode 100644 sodium/sodium_xmm6int_u4.h create mode 100644 sodium/sodium_xmm6int_u8.h delete mode 100644 windows/mbedtls/aes.h delete mode 100644 windows/mbedtls/aria.h delete mode 100644 windows/mbedtls/asn1.h delete mode 100644 windows/mbedtls/asn1write.h delete mode 100644 windows/mbedtls/base64.h delete mode 100644 windows/mbedtls/bignum.h delete mode 100644 windows/mbedtls/build_info.h delete mode 100644 windows/mbedtls/camellia.h delete mode 100644 windows/mbedtls/ccm.h delete mode 100644 windows/mbedtls/chacha20.h delete mode 100644 windows/mbedtls/chachapoly.h delete mode 100644 windows/mbedtls/check_config.h delete mode 100644 windows/mbedtls/cipher.h delete mode 100644 windows/mbedtls/cmac.h delete mode 100644 windows/mbedtls/compat-2.x.h delete mode 100644 windows/mbedtls/config_psa.h delete mode 100644 windows/mbedtls/ctr_drbg.h delete mode 100644 windows/mbedtls/debug.h delete mode 100644 windows/mbedtls/des.h delete mode 100644 windows/mbedtls/dhm.h delete mode 100644 windows/mbedtls/ecdh.h delete mode 100644 windows/mbedtls/ecdsa.h delete mode 100644 windows/mbedtls/ecjpake.h delete mode 100644 windows/mbedtls/ecp.h delete mode 100644 windows/mbedtls/entropy.h delete mode 100644 windows/mbedtls/error.h delete mode 100644 windows/mbedtls/gcm.h delete mode 100644 windows/mbedtls/hkdf.h delete mode 100644 windows/mbedtls/hmac_drbg.h delete mode 100644 windows/mbedtls/mbedtls_config.h delete mode 100644 windows/mbedtls/md.h delete mode 100644 windows/mbedtls/md5.h delete mode 100644 windows/mbedtls/memory_buffer_alloc.h delete mode 100644 windows/mbedtls/net_sockets.h delete mode 100644 windows/mbedtls/nist_kw.h delete mode 100644 windows/mbedtls/oid.h delete mode 100644 windows/mbedtls/pem.h delete mode 100644 windows/mbedtls/pk.h delete mode 100644 windows/mbedtls/pkcs12.h delete mode 100644 windows/mbedtls/pkcs5.h delete mode 100644 windows/mbedtls/platform.h delete mode 100644 windows/mbedtls/platform_time.h delete mode 100644 windows/mbedtls/platform_util.h delete mode 100644 windows/mbedtls/poly1305.h delete mode 100644 windows/mbedtls/private_access.h delete mode 100644 windows/mbedtls/psa_util.h delete mode 100644 windows/mbedtls/ripemd160.h delete mode 100644 windows/mbedtls/rsa.h delete mode 100644 windows/mbedtls/sha1.h delete mode 100644 windows/mbedtls/sha256.h delete mode 100644 windows/mbedtls/sha512.h delete mode 100644 windows/mbedtls/ssl.h delete mode 100644 windows/mbedtls/ssl_cache.h delete mode 100644 windows/mbedtls/ssl_ciphersuites.h delete mode 100644 windows/mbedtls/ssl_cookie.h delete mode 100644 windows/mbedtls/ssl_ticket.h delete mode 100644 windows/mbedtls/threading.h delete mode 100644 windows/mbedtls/timing.h delete mode 100644 windows/mbedtls/version.h delete mode 100644 windows/mbedtls/x509.h delete mode 100644 windows/mbedtls/x509_crl.h delete mode 100644 windows/mbedtls/x509_crt.h delete mode 100644 windows/mbedtls/x509_csr.h delete mode 100644 windows/sodium.h delete mode 100644 windows/sodium.lib delete mode 100644 windows/sodium/crypto_aead_aes256gcm.h delete mode 100644 windows/sodium/crypto_auth_hmacsha256.h delete mode 100644 windows/sodium/crypto_hash_sha256.h delete mode 100644 windows/sodium/crypto_kdf.h delete mode 100644 windows/sodium/crypto_kdf_blake2b.h delete mode 100644 windows/sodium/crypto_pwhash.h delete mode 100644 windows/sodium/crypto_pwhash_argon2i.h delete mode 100644 windows/sodium/crypto_pwhash_argon2id.h delete mode 100644 windows/sodium/crypto_pwhash_scryptsalsa208sha256.h delete mode 100644 windows/sodium/crypto_stream_aes128ctr.h delete mode 100644 windows/sodium/export.h diff --git a/certs.c b/certs/certs.c similarity index 100% rename from certs.c rename to certs/certs.c diff --git a/certs.h b/certs/certs.h similarity index 100% rename from certs.h rename to certs/certs.h diff --git a/netcode.io b/netcode.io deleted file mode 160000 index 8849a752..00000000 --- a/netcode.io +++ /dev/null @@ -1 +0,0 @@ -Subproject commit 8849a7523819f471e2c9fccb947b400e1593a034 diff --git a/netcode/netcode.c b/netcode/netcode.c new file mode 100755 index 00000000..230c599a --- /dev/null +++ b/netcode/netcode.c @@ -0,0 +1,8963 @@ +/* + netcode reference implementation + + Copyright © 2017 - 2024, Mas Bandwidth LLC + + Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: + + 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the distribution. + + 3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived + from this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, + INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, + WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE + USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ + +#include "netcode.h" +#include +#include +#include +#include +#include +#include +#include + +#ifdef _MSC_VER +#define SODIUM_STATIC +#pragma warning(disable:4996) +#endif // #ifdef _MSC_VER + +#include + +#define NETCODE_SOCKET_IPV6 1 +#define NETCODE_SOCKET_IPV4 2 + +#define NETCODE_CONNECT_TOKEN_NONCE_BYTES 24 +#define NETCODE_CONNECT_TOKEN_PRIVATE_BYTES 1024 +#define NETCODE_CHALLENGE_TOKEN_BYTES 300 +#define NETCODE_VERSION_INFO_BYTES 13 +#define NETCODE_MAX_PACKET_BYTES 1300 +#define NETCODE_MAX_PAYLOAD_BYTES 1200 +#define NETCODE_MAX_ADDRESS_STRING_LENGTH 256 +#define NETCODE_PACKET_QUEUE_SIZE 256 +#define NETCODE_REPLAY_PROTECTION_BUFFER_SIZE 256 +#define NETCODE_CLIENT_MAX_RECEIVE_PACKETS 64 +#define NETCODE_SERVER_MAX_RECEIVE_PACKETS ( 64 * NETCODE_MAX_CLIENTS ) +#define NETCODE_CLIENT_SOCKET_SNDBUF_SIZE ( 256 * 1024 ) +#define NETCODE_CLIENT_SOCKET_RCVBUF_SIZE ( 256 * 1024 ) +#define NETCODE_SERVER_SOCKET_SNDBUF_SIZE ( 4 * 1024 * 1024 ) +#define NETCODE_SERVER_SOCKET_RCVBUF_SIZE ( 4 * 1024 * 1024 ) + +#define NETCODE_VERSION_INFO ( (uint8_t*) "NETCODE 1.02" ) +#define NETCODE_PACKET_SEND_RATE 10.0 +#define NETCODE_NUM_DISCONNECT_PACKETS 10 + +#ifndef NETCODE_ENABLE_TESTS +#define NETCODE_ENABLE_TESTS 0 +#endif // #ifndef NETCODE_ENABLE_TESTS + +#ifndef NETCODE_ENABLE_LOGGING +#define NETCODE_ENABLE_LOGGING 1 +#endif // #ifndef NETCODE_ENABLE_LOGGING + +// ------------------------------------------------------------------ + +static void netcode_default_assert_handler( NETCODE_CONST char * condition, NETCODE_CONST char * function, NETCODE_CONST char * file, int line ) +{ + printf( "assert failed: ( %s ), function %s, file %s, line %d\n", condition, function, file, line ); + #if defined( __GNUC__ ) + __builtin_trap(); + #elif defined( _MSC_VER ) + __debugbreak(); + #endif + exit( 1 ); +} + +static int log_level; +static int (*printf_function)( NETCODE_CONST char *, ... ) = ( int (*)( NETCODE_CONST char *, ... ) ) printf; +void (*netcode_assert_function)( NETCODE_CONST char *, NETCODE_CONST char *, NETCODE_CONST char * file, int line ) = netcode_default_assert_handler; + +void netcode_log_level( int level ) +{ + log_level = level; +} + +void netcode_set_printf_function( int (*function)( NETCODE_CONST char *, ... ) ) +{ + netcode_assert( function ); + printf_function = function; +} + +void netcode_set_assert_function( void (*function)( NETCODE_CONST char *, NETCODE_CONST char *, NETCODE_CONST char * file, int line ) ) +{ + netcode_assert_function = function; +} + +#if NETCODE_ENABLE_LOGGING + +void netcode_printf( int level, NETCODE_CONST char * format, ... ) +{ + if ( level > log_level ) + return; + va_list args; + va_start( args, format ); + char buffer[4*1024]; + vsnprintf( buffer, sizeof(buffer), format, args ); + printf_function( "%s", buffer ); + va_end( args ); +} + +#else // #if NETCODE_ENABLE_LOGGING + +void netcode_printf( int level, NETCODE_CONST char * format, ... ) +{ + (void) level; + (void) format; +} + +#endif // #if NETCODE_ENABLE_LOGGING + +void * netcode_default_allocate_function( void * context, uint64_t bytes ) +{ + (void) context; + return malloc( bytes ); +} + +void netcode_default_free_function( void * context, void * pointer ) +{ + (void) context; + free( pointer ); +} + +// ------------------------------------------------------------------ + +#if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + + #define NOMINMAX + #define _WINSOCK_DEPRECATED_NO_WARNINGS + #include + #include + #include + #include + #pragma comment( lib, "WS2_32.lib" ) + #pragma comment( lib, "IPHLPAPI.lib" ) + + #ifdef SetPort + #undef SetPort + #endif // #ifdef SetPort + + #include + #pragma comment( lib, "IPHLPAPI.lib" ) + +#elif NETCODE_PLATFORM == NETCODE_PLATFORM_MAC || NETCODE_PLATFORM == NETCODE_PLATFORM_UNIX + + #include + #include + #include + #include + #include + #include + #include + #include + #include + #include + #include + +#else + + #error netcode - unknown platform! + +#endif + +// ---------------------------------------------------------------- + +#ifdef __MINGW32__ + const char *inet_ntop(int af, const void *src, char *dst, socklen_t cnt) + { + if ( af == AF_INET ) + { + struct sockaddr_in in; + memset(&in, 0, sizeof(in)); + in.sin_family = AF_INET; + memcpy(&in.sin_addr, src, sizeof(struct in_addr)); + getnameinfo((struct sockaddr *)&in, sizeof(struct sockaddr_in), dst, cnt, NULL, 0, NI_NUMERICHOST); + return dst; + } + else if ( af == AF_INET6 ) + { + struct sockaddr_in6 in; + memset(&in, 0, sizeof(in)); + in.sin6_family = AF_INET6; + memcpy(&in.sin6_addr, src, sizeof(struct in_addr6)); + getnameinfo((struct sockaddr *)&in, sizeof(struct sockaddr_in6), dst, cnt, NULL, 0, NI_NUMERICHOST); + return dst; + } + + return NULL; + } + + #define NS_INADDRSZ 4 + #define NS_IN6ADDRSZ 16 + #define NS_INT16SZ 2 + + int inet_pton4(const char *src, char *dst) + { + uint8_t tmp[NS_INADDRSZ], *tp; + int saw_digit = 0; + int octets = 0; + *(tp = tmp) = 0; + int ch; + + while ((ch = *src++) != '\0') + { + if ( ch >= '0' && ch <= '9' ) + { + uint32_t n = *tp * 10 + (ch - '0'); + + if ( saw_digit && *tp == 0 ) + return 0; + if ( n > 255 ) + return 0; + *tp = n; + if ( !saw_digit ) + { + if ( ++octets > 4 ) + return 0; + saw_digit = 1; + } + } + else if ( ch == '.' && saw_digit ) + { + if ( octets == 4 ) + return 0; + *++tp = 0; + saw_digit = 0; + } + else + return 0; + } + if ( octets < 4 ) + return 0; + + memcpy(dst, tmp, NS_INADDRSZ); + + return 1; + } + + int inet_pton6(const char *src, char *dst) + { + static const char xdigits[] = "0123456789abcdef"; + uint8_t tmp[NS_IN6ADDRSZ]; + uint8_t *tp = (uint8_t*) memset(tmp, '\0', NS_IN6ADDRSZ); + uint8_t *endp = tp + NS_IN6ADDRSZ; + uint8_t *colonp = NULL; + + // Leading :: requires some special handling + if ( *src == ':' ) + { + if ( *++src != ':' ) + return 0; + } + + const char *curtok = src; + int saw_xdigit = 0; + uint32_t val = 0; + int ch; + + while ( (ch = tolower(*src++)) != '\0' ) + { + const char *pch = strchr(xdigits, ch); + if ( pch != NULL ) + { + val <<= 4; + val |= (pch - xdigits); + if ( val > 0xffff ) + return 0; + saw_xdigit = 1; + continue; + } + if ( ch == ':' ) + { + curtok = src; + if ( !saw_xdigit ) + { + if ( colonp ) + return 0; + colonp = tp; + continue; + } + else if ( *src == '\0' ) + { + return 0; + } + if ( tp + NS_INT16SZ > endp ) + return 0; + *tp++ = (uint8_t) (val >> 8) & 0xff; + *tp++ = (uint8_t) val & 0xff; + saw_xdigit = 0; + val = 0; + continue; + } + if ( ch == '.' && ((tp + NS_INADDRSZ) <= endp) && inet_pton4(curtok, (char*) tp) > 0 ) + { + tp += NS_INADDRSZ; + saw_xdigit = 0; + break; + } + return 0; + } + + if ( saw_xdigit ) + { + if ( tp + NS_INT16SZ > endp ) + return 0; + *tp++ = (uint8_t) (val >> 8) & 0xff; + *tp++ = (uint8_t) val & 0xff; + } + + if ( colonp != NULL ) + { + const int n = tp - colonp; + if ( tp == endp ) + return 0; + for (int i = 1; i <= n; i++) + { + endp[-i] = colonp[n - i]; + colonp[n - i] = 0; + } + tp = endp; + } + if ( tp != endp ) + return 0; + + memcpy(dst, tmp, NS_IN6ADDRSZ); + + return 1; + } + + int inet_pton(int af, const char *src, void *dst) + { + switch ( af ) + { + case AF_INET: + return inet_pton4(src, (char *) dst); + + case AF_INET6: + return inet_pton6(src, (char *) dst); + + default: + return -1; + } + } +#endif + +// ---------------------------------------------------------------- + +int netcode_parse_address( NETCODE_CONST char * address_string_in, struct netcode_address_t * address ) +{ + netcode_assert( address_string_in ); + netcode_assert( address ); + + memset( address, 0, sizeof( struct netcode_address_t ) ); + + // first try to parse the string as an IPv6 address: + // 1. if the first character is '[' then it's probably an ipv6 in form "[addr6]:portnum" + // 2. otherwise try to parse as a raw IPv6 address using inet_pton + + #define NETCODE_ADDRESS_BUFFER_SAFETY 32 + + char buffer[NETCODE_MAX_ADDRESS_STRING_LENGTH + NETCODE_ADDRESS_BUFFER_SAFETY*2]; + + char * address_string = buffer + NETCODE_ADDRESS_BUFFER_SAFETY; + strncpy( address_string, address_string_in, NETCODE_MAX_ADDRESS_STRING_LENGTH - 1 ); + address_string[NETCODE_MAX_ADDRESS_STRING_LENGTH-1] = '\0'; + + int address_string_length = (int) strlen( address_string ); + + if ( address_string[0] == '[' ) + { + int base_index = address_string_length - 1; + + int i; + for ( i = 0; i < 6; ++i ) // note: no need to search past 6 characters as ":65535" is longest possible port value + { + int index = base_index - i; + if ( index < 3 ) + return NETCODE_ERROR; + if ( address_string[index] == ':' ) + { + address->port = (uint16_t) ( atoi( &address_string[index + 1] ) ); + address_string[index-1] = '\0'; + } + } + address_string += 1; + } + + struct in6_addr sockaddr6; + if ( inet_pton( AF_INET6, address_string, &sockaddr6 ) == 1 ) + { + address->type = NETCODE_ADDRESS_IPV6; + int i; + for ( i = 0; i < 8; ++i ) + { + address->data.ipv6[i] = ntohs( ( (uint16_t*) &sockaddr6 ) [i] ); + } + return NETCODE_OK; + } + + // otherwise it's probably an IPv4 address: + // 1. look for ":portnum", if found save the portnum and strip it out + // 2. parse remaining ipv4 address via inet_pton + + address_string_length = (int) strlen( address_string ); + int base_index = address_string_length - 1; + int i; + for ( i = 0; i < 6; ++i ) + { + int index = base_index - i; + if ( index < 0 ) + break; + if ( address_string[index] == ':' ) + { + address->port = (uint16_t) atoi( &address_string[index+1] ); + address_string[index] = '\0'; + } + } + + struct sockaddr_in sockaddr4; + if ( inet_pton( AF_INET, address_string, &sockaddr4.sin_addr ) == 1 ) + { + address->type = NETCODE_ADDRESS_IPV4; + address->data.ipv4[3] = (uint8_t) ( ( sockaddr4.sin_addr.s_addr & 0xFF000000 ) >> 24 ); + address->data.ipv4[2] = (uint8_t) ( ( sockaddr4.sin_addr.s_addr & 0x00FF0000 ) >> 16 ); + address->data.ipv4[1] = (uint8_t) ( ( sockaddr4.sin_addr.s_addr & 0x0000FF00 ) >> 8 ); + address->data.ipv4[0] = (uint8_t) ( ( sockaddr4.sin_addr.s_addr & 0x000000FF ) ); + return NETCODE_OK; + } + + return NETCODE_ERROR; +} + +char * netcode_address_to_string( struct netcode_address_t * address, char * buffer ) +{ + netcode_assert( address ); + netcode_assert( buffer ); + + if ( address->type == NETCODE_ADDRESS_IPV6 ) + { + if ( address->port == 0 ) + { + uint16_t ipv6_network_order[8]; + int i; + for ( i = 0; i < 8; ++i ) + ipv6_network_order[i] = htons( address->data.ipv6[i] ); + inet_ntop( AF_INET6, (void*) ipv6_network_order, buffer, NETCODE_MAX_ADDRESS_STRING_LENGTH ); + return buffer; + } + else + { + char address_string[INET6_ADDRSTRLEN]; + uint16_t ipv6_network_order[8]; + int i; + for ( i = 0; i < 8; ++i ) + ipv6_network_order[i] = htons( address->data.ipv6[i] ); + inet_ntop( AF_INET6, (void*) ipv6_network_order, address_string, INET6_ADDRSTRLEN ); + snprintf( buffer, NETCODE_MAX_ADDRESS_STRING_LENGTH, "[%s]:%d", address_string, address->port ); + return buffer; + } + } + else if ( address->type == NETCODE_ADDRESS_IPV4 ) + { + if ( address->port != 0 ) + { + snprintf( buffer, NETCODE_MAX_ADDRESS_STRING_LENGTH, "%d.%d.%d.%d:%d", + address->data.ipv4[0], + address->data.ipv4[1], + address->data.ipv4[2], + address->data.ipv4[3], + address->port ); + } + else + { + snprintf( buffer, NETCODE_MAX_ADDRESS_STRING_LENGTH, "%d.%d.%d.%d", + address->data.ipv4[0], + address->data.ipv4[1], + address->data.ipv4[2], + address->data.ipv4[3] ); + } + return buffer; + } + else + { + snprintf( buffer, NETCODE_MAX_ADDRESS_STRING_LENGTH, "%s", "NONE" ); + return buffer; + } +} + +int netcode_address_equal( struct netcode_address_t * a, struct netcode_address_t * b ) +{ + netcode_assert( a ); + netcode_assert( b ); + + if ( a->type != b->type ) + return 0; + + if ( a->port != b->port ) + return 0; + + if ( a->type == NETCODE_ADDRESS_IPV4 ) + { + int i; + for ( i = 0; i < 4; ++i ) + { + if ( a->data.ipv4[i] != b->data.ipv4[i] ) + return 0; + } + } + else if ( a->type == NETCODE_ADDRESS_IPV6 ) + { + int i; + for ( i = 0; i < 8; ++i ) + { + if ( a->data.ipv6[i] != b->data.ipv6[i] ) + return 0; + } + } + else + { + return 0; + } + + return 1; +} + +// ---------------------------------------------------------------- + +struct netcode_t +{ + int initialized; +}; + +static struct netcode_t netcode; + +int netcode_init() +{ + netcode_assert( !netcode.initialized ); + +#if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + WSADATA WsaData; + if ( WSAStartup( MAKEWORD(2,2), &WsaData ) != NO_ERROR ) + return NETCODE_ERROR; +#endif // #if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + + if ( sodium_init() == -1 ) + return NETCODE_ERROR; + + netcode.initialized = 1; + + return NETCODE_OK; +} + +void netcode_term() +{ + netcode_assert( netcode.initialized ); + +#if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + WSACleanup(); +#endif // #if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + + netcode.initialized = 0; +} + +// ---------------------------------------------------------------- + +#if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS +typedef uint64_t netcode_socket_handle_t; +#else // #if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS +typedef int netcode_socket_handle_t; +#endif // #if NETCODE_PLATFORM == NETCODe_PLATFORM_WINDOWS + +struct netcode_socket_t +{ + struct netcode_address_t address; + netcode_socket_handle_t handle; +}; + + +struct netcode_socket_holder_t +{ + struct netcode_socket_t ipv4; + struct netcode_socket_t ipv6; +}; + +#define NETCODE_SOCKET_ERROR_NONE 0 +#define NETCODE_SOCKET_ERROR_CREATE_FAILED 1 +#define NETCODE_SOCKET_ERROR_SET_NON_BLOCKING_FAILED 2 +#define NETCODE_SOCKET_ERROR_SOCKOPT_IPV6_ONLY_FAILED 3 +#define NETCODE_SOCKET_ERROR_SOCKOPT_RCVBUF_FAILED 4 +#define NETCODE_SOCKET_ERROR_SOCKOPT_SNDBUF_FAILED 5 +#define NETCODE_SOCKET_ERROR_BIND_IPV4_FAILED 6 +#define NETCODE_SOCKET_ERROR_BIND_IPV6_FAILED 7 +#define NETCODE_SOCKET_ERROR_GET_SOCKNAME_IPV4_FAILED 8 +#define NETCODE_SOCKET_ERROR_GET_SOCKNAME_IPV6_FAILED 7 + +void netcode_socket_destroy( struct netcode_socket_t * socket ) +{ + netcode_assert( socket ); + netcode_assert( netcode.initialized ); + + if ( socket->handle != 0 ) + { + #if NETCODE_PLATFORM == NETCODE_PLATFORM_MAC || NETCODE_PLATFORM == NETCODE_PLATFORM_UNIX + close( socket->handle ); + #elif NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + closesocket( socket->handle ); + #else + #error unsupported platform + #endif + socket->handle = 0; + } +} + +int netcode_socket_create( struct netcode_socket_t * s, struct netcode_address_t * address, int send_buffer_size, int receive_buffer_size ) +{ + netcode_assert( s ); + netcode_assert( address ); + netcode_assert( netcode.initialized ); + + netcode_assert( address->type != NETCODE_ADDRESS_NONE ); + + s->address = *address; + + // create socket + + s->handle = socket( ( address->type == NETCODE_ADDRESS_IPV6 ) ? AF_INET6 : AF_INET, SOCK_DGRAM, IPPROTO_UDP ); + +#if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + if ( s->handle == INVALID_SOCKET ) +#else // #if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + if ( s->handle <= 0 ) +#endif // #if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to create socket\n" ); + return NETCODE_SOCKET_ERROR_CREATE_FAILED; + } + + // IMPORTANT: tell windows we don't want to receive any connection reset messages + // for this socket, otherwise recvfrom errors out when client sockets disconnect hard + // in response to ICMP messages. +#if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + #define SIO_UDP_CONNRESET _WSAIOW(IOC_VENDOR, 12) + BOOL bNewBehavior = FALSE; + DWORD dwBytesReturned = 0; + WSAIoctl( s->handle, SIO_UDP_CONNRESET, &bNewBehavior, sizeof(bNewBehavior), NULL, 0, &dwBytesReturned, NULL, NULL ); +#endif + + // force IPv6 only if necessary + + if ( address->type == NETCODE_ADDRESS_IPV6 ) + { + int yes = 1; + if ( setsockopt( s->handle, IPPROTO_IPV6, IPV6_V6ONLY, (char*)&yes, sizeof(yes) ) != 0 ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to set socket ipv6 only\n" ); + netcode_socket_destroy( s ); + return NETCODE_SOCKET_ERROR_SOCKOPT_IPV6_ONLY_FAILED; + } + } + + // increase socket send and receive buffer sizes + + if ( setsockopt( s->handle, SOL_SOCKET, SO_SNDBUF, (char*)&send_buffer_size, sizeof(int) ) != 0 ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to set socket send buffer size\n" ); + netcode_socket_destroy( s ); + return NETCODE_SOCKET_ERROR_SOCKOPT_SNDBUF_FAILED; + } + + if ( setsockopt( s->handle, SOL_SOCKET, SO_RCVBUF, (char*)&receive_buffer_size, sizeof(int) ) != 0 ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to set socket receive buffer size\n" ); + netcode_socket_destroy( s ); + return NETCODE_SOCKET_ERROR_SOCKOPT_RCVBUF_FAILED; + } + + // bind to port + + if ( address->type == NETCODE_ADDRESS_IPV6 ) + { + struct sockaddr_in6 socket_address; + memset( &socket_address, 0, sizeof( struct sockaddr_in6 ) ); + socket_address.sin6_family = AF_INET6; + int i; + for ( i = 0; i < 8; ++i ) + { + ( (uint16_t*) &socket_address.sin6_addr ) [i] = htons( address->data.ipv6[i] ); + } + socket_address.sin6_port = htons( address->port ); + + if ( bind( s->handle, (struct sockaddr*) &socket_address, sizeof( socket_address ) ) < 0 ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to bind socket (ipv6)\n" ); + netcode_socket_destroy( s ); + return NETCODE_SOCKET_ERROR_BIND_IPV6_FAILED; + } + } + else + { + struct sockaddr_in socket_address; + memset( &socket_address, 0, sizeof( socket_address ) ); + socket_address.sin_family = AF_INET; + socket_address.sin_addr.s_addr = ( ( (uint32_t) address->data.ipv4[0] ) ) | + ( ( (uint32_t) address->data.ipv4[1] ) << 8 ) | + ( ( (uint32_t) address->data.ipv4[2] ) << 16 ) | + ( ( (uint32_t) address->data.ipv4[3] ) << 24 ); + socket_address.sin_port = htons( address->port ); + + if ( bind( s->handle, (struct sockaddr*) &socket_address, sizeof( socket_address ) ) < 0 ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to bind socket (ipv4)\n" ); + netcode_socket_destroy( s ); + return NETCODE_SOCKET_ERROR_BIND_IPV4_FAILED; + } + } + + // if bound to port 0 find the actual port we got + + if ( address->port == 0 ) + { + if ( address->type == NETCODE_ADDRESS_IPV6 ) + { + struct sockaddr_in6 sin; + socklen_t len = sizeof( sin ); + if ( getsockname( s->handle, (struct sockaddr*)&sin, &len ) == -1 ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to get socket port (ipv6)\n" ); + netcode_socket_destroy( s ); + return NETCODE_SOCKET_ERROR_GET_SOCKNAME_IPV6_FAILED; + } + s->address.port = ntohs( sin.sin6_port ); + } + else + { + struct sockaddr_in sin; + socklen_t len = sizeof( sin ); + if ( getsockname( s->handle, (struct sockaddr*)&sin, &len ) == -1 ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to get socket port (ipv4)\n" ); + netcode_socket_destroy( s ); + return NETCODE_SOCKET_ERROR_GET_SOCKNAME_IPV4_FAILED; + } + s->address.port = ntohs( sin.sin_port ); + } + } + + // set non-blocking io + +#if NETCODE_PLATFORM == NETCODE_PLATFORM_MAC || NETCODE_PLATFORM == NETCODE_PLATFORM_UNIX + + int non_blocking = 1; + if ( fcntl( s->handle, F_SETFL, O_NONBLOCK, non_blocking ) == -1 ) + { + netcode_socket_destroy( s ); + return NETCODE_SOCKET_ERROR_SET_NON_BLOCKING_FAILED; + } + +#elif NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + + DWORD nonBlocking = 1; + if ( ioctlsocket( s->handle, FIONBIO, &nonBlocking ) != 0 ) + { + netcode_socket_destroy( s ); + return NETCODE_SOCKET_ERROR_SET_NON_BLOCKING_FAILED; + } + +#else + + #error unsupported platform + +#endif + + return NETCODE_SOCKET_ERROR_NONE; +} + +void netcode_socket_send_packet( struct netcode_socket_t * socket, struct netcode_address_t * to, void * packet_data, int packet_bytes ) +{ + netcode_assert( socket ); + netcode_assert( socket->handle != 0 ); + netcode_assert( to ); + netcode_assert( to->type == NETCODE_ADDRESS_IPV6 || to->type == NETCODE_ADDRESS_IPV4 ); + netcode_assert( packet_data ); + netcode_assert( packet_bytes > 0 ); + + if ( to->type == NETCODE_ADDRESS_IPV6 ) + { + struct sockaddr_in6 socket_address; + memset( &socket_address, 0, sizeof( socket_address ) ); + socket_address.sin6_family = AF_INET6; + int i; + for ( i = 0; i < 8; ++i ) + { + ( (uint16_t*) &socket_address.sin6_addr ) [i] = htons( to->data.ipv6[i] ); + } + socket_address.sin6_port = htons( to->port ); + int result = sendto( socket->handle, (char*) packet_data, packet_bytes, 0, (struct sockaddr*) &socket_address, sizeof( struct sockaddr_in6 ) ); + (void) result; + } + else if ( to->type == NETCODE_ADDRESS_IPV4 ) + { + struct sockaddr_in socket_address; + memset( &socket_address, 0, sizeof( socket_address ) ); + socket_address.sin_family = AF_INET; + socket_address.sin_addr.s_addr = ( ( (uint32_t) to->data.ipv4[0] ) ) | + ( ( (uint32_t) to->data.ipv4[1] ) << 8 ) | + ( ( (uint32_t) to->data.ipv4[2] ) << 16 ) | + ( ( (uint32_t) to->data.ipv4[3] ) << 24 ); + socket_address.sin_port = htons( to->port ); + int result = sendto( socket->handle, (NETCODE_CONST char*) packet_data, packet_bytes, 0, (struct sockaddr*) &socket_address, sizeof( struct sockaddr_in ) ); + (void) result; + } +} + +int netcode_socket_receive_packet( struct netcode_socket_t * socket, struct netcode_address_t * from, void * packet_data, int max_packet_size ) +{ + netcode_assert( socket ); + netcode_assert( socket->handle != 0 ); + netcode_assert( from ); + netcode_assert( packet_data ); + netcode_assert( max_packet_size > 0 ); + +#if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + typedef int socklen_t; +#endif // #if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + + struct sockaddr_storage sockaddr_from; + socklen_t from_length = sizeof( sockaddr_from ); + + int result = recvfrom( socket->handle, (char*) packet_data, max_packet_size, 0, (struct sockaddr*) &sockaddr_from, &from_length ); + +#if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + if ( result == SOCKET_ERROR ) + { + int error = WSAGetLastError(); + + if ( error == WSAEWOULDBLOCK ) + return 0; + + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: recvfrom failed with error %d\n", error ); + + return 0; + } +#else // #if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + if ( result <= 0 ) + { + if ( errno == EAGAIN ) + return 0; + + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: recvfrom failed with error %d\n", errno ); + + return 0; + } +#endif // #if NETCODE_PLATFORM == NETCODE_PLATFORM_WINDOWS + + if ( sockaddr_from.ss_family == AF_INET6 ) + { + struct sockaddr_in6 * addr_ipv6 = (struct sockaddr_in6*) &sockaddr_from; + from->type = NETCODE_ADDRESS_IPV6; + int i; + for ( i = 0; i < 8; ++i ) + { + from->data.ipv6[i] = ntohs( ( (uint16_t*) &addr_ipv6->sin6_addr ) [i] ); + } + from->port = ntohs( addr_ipv6->sin6_port ); + } + else if ( sockaddr_from.ss_family == AF_INET ) + { + struct sockaddr_in * addr_ipv4 = (struct sockaddr_in*) &sockaddr_from; + from->type = NETCODE_ADDRESS_IPV4; + from->data.ipv4[0] = (uint8_t) ( ( addr_ipv4->sin_addr.s_addr & 0x000000FF ) ); + from->data.ipv4[1] = (uint8_t) ( ( addr_ipv4->sin_addr.s_addr & 0x0000FF00 ) >> 8 ); + from->data.ipv4[2] = (uint8_t) ( ( addr_ipv4->sin_addr.s_addr & 0x00FF0000 ) >> 16 ); + from->data.ipv4[3] = (uint8_t) ( ( addr_ipv4->sin_addr.s_addr & 0xFF000000 ) >> 24 ); + from->port = ntohs( addr_ipv4->sin_port ); + } + else + { + netcode_assert( 0 ); + return 0; + } + + netcode_assert( result >= 0 ); + + int bytes_read = result; + + return bytes_read; +} + +// ---------------------------------------------------------------- + +void netcode_write_uint8( uint8_t ** p, uint8_t value ) +{ + **p = value; + ++(*p); +} + +void netcode_write_uint16( uint8_t ** p, uint16_t value ) +{ + (*p)[0] = value & 0xFF; + (*p)[1] = value >> 8; + *p += 2; +} + +void netcode_write_uint32( uint8_t ** p, uint32_t value ) +{ + (*p)[0] = value & 0xFF; + (*p)[1] = ( value >> 8 ) & 0xFF; + (*p)[2] = ( value >> 16 ) & 0xFF; + (*p)[3] = value >> 24; + *p += 4; +} + +void netcode_write_uint64( uint8_t ** p, uint64_t value ) +{ + (*p)[0] = value & 0xFF; + (*p)[1] = ( value >> 8 ) & 0xFF; + (*p)[2] = ( value >> 16 ) & 0xFF; + (*p)[3] = ( value >> 24 ) & 0xFF; + (*p)[4] = ( value >> 32 ) & 0xFF; + (*p)[5] = ( value >> 40 ) & 0xFF; + (*p)[6] = ( value >> 48 ) & 0xFF; + (*p)[7] = value >> 56; + *p += 8; +} + +void netcode_write_bytes( uint8_t ** p, uint8_t * byte_array, int num_bytes ) +{ + int i; + for ( i = 0; i < num_bytes; ++i ) + { + netcode_write_uint8( p, byte_array[i] ); + } +} + +uint8_t netcode_read_uint8( uint8_t ** p ) +{ + uint8_t value = **p; + ++(*p); + return value; +} + +uint16_t netcode_read_uint16( uint8_t ** p ) +{ + uint16_t value; + value = (*p)[0]; + value |= ( ( (uint16_t)( (*p)[1] ) ) << 8 ); + *p += 2; + return value; +} + +uint32_t netcode_read_uint32( uint8_t ** p ) +{ + uint32_t value; + value = (*p)[0]; + value |= ( ( (uint32_t)( (*p)[1] ) ) << 8 ); + value |= ( ( (uint32_t)( (*p)[2] ) ) << 16 ); + value |= ( ( (uint32_t)( (*p)[3] ) ) << 24 ); + *p += 4; + return value; +} + +uint64_t netcode_read_uint64( uint8_t ** p ) +{ + uint64_t value; + value = (*p)[0]; + value |= ( ( (uint64_t)( (*p)[1] ) ) << 8 ); + value |= ( ( (uint64_t)( (*p)[2] ) ) << 16 ); + value |= ( ( (uint64_t)( (*p)[3] ) ) << 24 ); + value |= ( ( (uint64_t)( (*p)[4] ) ) << 32 ); + value |= ( ( (uint64_t)( (*p)[5] ) ) << 40 ); + value |= ( ( (uint64_t)( (*p)[6] ) ) << 48 ); + value |= ( ( (uint64_t)( (*p)[7] ) ) << 56 ); + *p += 8; + return value; +} + +void netcode_read_bytes( uint8_t ** p, uint8_t * byte_array, int num_bytes ) +{ + int i; + for ( i = 0; i < num_bytes; ++i ) + { + byte_array[i] = netcode_read_uint8( p ); + } +} + +// ---------------------------------------------------------------- + +void netcode_generate_key( uint8_t * key ) +{ + netcode_assert( key ); + randombytes_buf( key, NETCODE_KEY_BYTES ); +} + +void netcode_generate_nonce( uint8_t * nonce ) +{ + netcode_assert( nonce ); + randombytes_buf( nonce, NETCODE_CONNECT_TOKEN_NONCE_BYTES ); +} + +void netcode_random_bytes( uint8_t * data, int bytes ) +{ + netcode_assert( data ); + netcode_assert( bytes > 0 ); + randombytes_buf( data, bytes ); +} + +int netcode_encrypt_aead_bignonce( uint8_t * message, uint64_t message_length, + uint8_t * additional, uint64_t additional_length, + NETCODE_CONST uint8_t * nonce, + NETCODE_CONST uint8_t * key ) +{ + unsigned long long encrypted_length; + + int result = crypto_aead_xchacha20poly1305_ietf_encrypt( message, &encrypted_length, + message, (unsigned long long) message_length, + additional, (unsigned long long) additional_length, + NULL, nonce, key ); + + if ( result != 0 ) + return NETCODE_ERROR; + + netcode_assert( encrypted_length == message_length + NETCODE_MAC_BYTES ); + + return NETCODE_OK; +} + +int netcode_decrypt_aead_bignonce( uint8_t * message, uint64_t message_length, + uint8_t * additional, uint64_t additional_length, + uint8_t * nonce, + uint8_t * key ) +{ + unsigned long long decrypted_length; + + int result = crypto_aead_xchacha20poly1305_ietf_decrypt( message, &decrypted_length, + NULL, + message, (unsigned long long) message_length, + additional, (unsigned long long) additional_length, + nonce, key ); + + if ( result != 0 ) + return NETCODE_ERROR; + + netcode_assert( decrypted_length == message_length - NETCODE_MAC_BYTES ); + + return NETCODE_OK; +} + +int netcode_encrypt_aead( uint8_t * message, uint64_t message_length, + uint8_t * additional, uint64_t additional_length, + NETCODE_CONST uint8_t * nonce, + NETCODE_CONST uint8_t * key ) +{ + unsigned long long encrypted_length; + + int result = crypto_aead_chacha20poly1305_ietf_encrypt( message, &encrypted_length, + message, (unsigned long long) message_length, + additional, (unsigned long long) additional_length, + NULL, nonce, key ); + + if ( result != 0 ) + return NETCODE_ERROR; + + netcode_assert( encrypted_length == message_length + NETCODE_MAC_BYTES ); + + return NETCODE_OK; +} + +int netcode_decrypt_aead( uint8_t * message, uint64_t message_length, + uint8_t * additional, uint64_t additional_length, + uint8_t * nonce, + uint8_t * key ) +{ + unsigned long long decrypted_length; + + int result = crypto_aead_chacha20poly1305_ietf_decrypt( message, &decrypted_length, + NULL, + message, (unsigned long long) message_length, + additional, (unsigned long long) additional_length, + nonce, key ); + + if ( result != 0 ) + return NETCODE_ERROR; + + netcode_assert( decrypted_length == message_length - NETCODE_MAC_BYTES ); + + return NETCODE_OK; +} + +// ---------------------------------------------------------------- + +struct netcode_connect_token_private_t +{ + uint64_t client_id; + int timeout_seconds; + int num_server_addresses; + struct netcode_address_t server_addresses[NETCODE_MAX_SERVERS_PER_CONNECT]; + uint8_t client_to_server_key[NETCODE_KEY_BYTES]; + uint8_t server_to_client_key[NETCODE_KEY_BYTES]; + uint8_t user_data[NETCODE_USER_DATA_BYTES]; +}; + +void netcode_generate_connect_token_private( struct netcode_connect_token_private_t * connect_token, + uint64_t client_id, + int timeout_seconds, + int num_server_addresses, + struct netcode_address_t * server_addresses, + uint8_t * user_data ) +{ + netcode_assert( connect_token ); + netcode_assert( num_server_addresses > 0 ); + netcode_assert( num_server_addresses <= NETCODE_MAX_SERVERS_PER_CONNECT ); + netcode_assert( server_addresses ); + netcode_assert( user_data ); + + connect_token->client_id = client_id; + connect_token->timeout_seconds = timeout_seconds; + connect_token->num_server_addresses = num_server_addresses; + + int i; + for ( i = 0; i < num_server_addresses; ++i ) + { + memcpy( &connect_token->server_addresses[i], &server_addresses[i], sizeof( struct netcode_address_t ) ); + } + + netcode_generate_key( connect_token->client_to_server_key ); + netcode_generate_key( connect_token->server_to_client_key ); + + if ( user_data != NULL ) + { + memcpy( connect_token->user_data, user_data, NETCODE_USER_DATA_BYTES ); + } + else + { + memset( connect_token->user_data, 0, NETCODE_USER_DATA_BYTES ); + } +} + +void netcode_write_connect_token_private( struct netcode_connect_token_private_t * connect_token, uint8_t * buffer, int buffer_length ) +{ + (void) buffer_length; + + netcode_assert( connect_token ); + netcode_assert( connect_token->num_server_addresses > 0 ); + netcode_assert( connect_token->num_server_addresses <= NETCODE_MAX_SERVERS_PER_CONNECT ); + netcode_assert( buffer ); + netcode_assert( buffer_length >= NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + uint8_t * start = buffer; + + (void) start; + + netcode_write_uint64( &buffer, connect_token->client_id ); + + netcode_write_uint32( &buffer, connect_token->timeout_seconds ); + + netcode_write_uint32( &buffer, connect_token->num_server_addresses ); + + int i,j; + + for ( i = 0; i < connect_token->num_server_addresses; ++i ) + { + if ( connect_token->server_addresses[i].type == NETCODE_ADDRESS_IPV4 ) + { + netcode_write_uint8( &buffer, NETCODE_ADDRESS_IPV4 ); + for ( j = 0; j < 4; ++j ) + { + netcode_write_uint8( &buffer, connect_token->server_addresses[i].data.ipv4[j] ); + } + netcode_write_uint16( &buffer, connect_token->server_addresses[i].port ); + } + else if ( connect_token->server_addresses[i].type == NETCODE_ADDRESS_IPV6 ) + { + netcode_write_uint8( &buffer, NETCODE_ADDRESS_IPV6 ); + for ( j = 0; j < 8; ++j ) + { + netcode_write_uint16( &buffer, connect_token->server_addresses[i].data.ipv6[j] ); + } + netcode_write_uint16( &buffer, connect_token->server_addresses[i].port ); + } + else + { + netcode_assert( 0 ); + } + } + + netcode_write_bytes( &buffer, connect_token->client_to_server_key, NETCODE_KEY_BYTES ); + + netcode_write_bytes( &buffer, connect_token->server_to_client_key, NETCODE_KEY_BYTES ); + + netcode_write_bytes( &buffer, connect_token->user_data, NETCODE_USER_DATA_BYTES ); + + netcode_assert( buffer - start <= NETCODE_CONNECT_TOKEN_PRIVATE_BYTES - NETCODE_MAC_BYTES ); + + memset( buffer, 0, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES - ( buffer - start ) ); +} + +int netcode_encrypt_connect_token_private( uint8_t * buffer, + int buffer_length, + uint8_t * version_info, + uint64_t protocol_id, + uint64_t expire_timestamp, + NETCODE_CONST uint8_t * nonce, + NETCODE_CONST uint8_t * key ) +{ + netcode_assert( buffer ); + netcode_assert( buffer_length == NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + netcode_assert( key ); + + (void) buffer_length; + + uint8_t additional_data[NETCODE_VERSION_INFO_BYTES+8+8]; + { + uint8_t * p = additional_data; + netcode_write_bytes( &p, version_info, NETCODE_VERSION_INFO_BYTES ); + netcode_write_uint64( &p, protocol_id ); + netcode_write_uint64( &p, expire_timestamp ); + } + + return netcode_encrypt_aead_bignonce( buffer, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES - NETCODE_MAC_BYTES, additional_data, sizeof( additional_data ), nonce, key ); +} + +int netcode_decrypt_connect_token_private( uint8_t * buffer, + int buffer_length, + uint8_t * version_info, + uint64_t protocol_id, + uint64_t expire_timestamp, + uint8_t * nonce, + uint8_t * key ) +{ + netcode_assert( buffer ); + netcode_assert( buffer_length == NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + netcode_assert( key ); + + (void) buffer_length; + + uint8_t additional_data[NETCODE_VERSION_INFO_BYTES+8+8]; + { + uint8_t * p = additional_data; + netcode_write_bytes( &p, version_info, NETCODE_VERSION_INFO_BYTES ); + netcode_write_uint64( &p, protocol_id ); + netcode_write_uint64( &p, expire_timestamp ); + } + return netcode_decrypt_aead_bignonce( buffer, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES, additional_data, sizeof( additional_data ), nonce, key ); +} + +int netcode_read_connect_token_private( uint8_t * buffer, int buffer_length, struct netcode_connect_token_private_t * connect_token ) +{ + netcode_assert( buffer ); + netcode_assert( connect_token ); + + if ( buffer_length < NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ) + return NETCODE_ERROR; + + connect_token->client_id = netcode_read_uint64( &buffer ); + + connect_token->timeout_seconds = (int) netcode_read_uint32( &buffer ); + + connect_token->num_server_addresses = netcode_read_uint32( &buffer ); + + if ( connect_token->num_server_addresses <= 0 ) + return NETCODE_ERROR; + + if ( connect_token->num_server_addresses > NETCODE_MAX_SERVERS_PER_CONNECT ) + return NETCODE_ERROR; + + int i,j; + + for ( i = 0; i < connect_token->num_server_addresses; ++i ) + { + connect_token->server_addresses[i].type = netcode_read_uint8( &buffer ); + + if ( connect_token->server_addresses[i].type == NETCODE_ADDRESS_IPV4 ) + { + for ( j = 0; j < 4; ++j ) + { + connect_token->server_addresses[i].data.ipv4[j] = netcode_read_uint8( &buffer ); + } + connect_token->server_addresses[i].port = netcode_read_uint16( &buffer ); + } + else if ( connect_token->server_addresses[i].type == NETCODE_ADDRESS_IPV6 ) + { + for ( j = 0; j < 8; ++j ) + { + connect_token->server_addresses[i].data.ipv6[j] = netcode_read_uint16( &buffer ); + } + connect_token->server_addresses[i].port = netcode_read_uint16( &buffer ); + } + else + { + return NETCODE_ERROR; + } + } + + netcode_read_bytes( &buffer, connect_token->client_to_server_key, NETCODE_KEY_BYTES ); + + netcode_read_bytes( &buffer, connect_token->server_to_client_key, NETCODE_KEY_BYTES ); + + netcode_read_bytes( &buffer, connect_token->user_data, NETCODE_USER_DATA_BYTES ); + + return NETCODE_OK; +} + +// ----------------------------------------------- + +struct netcode_challenge_token_t +{ + uint64_t client_id; + uint8_t user_data[NETCODE_USER_DATA_BYTES]; +}; + +void netcode_write_challenge_token( struct netcode_challenge_token_t * challenge_token, uint8_t * buffer, int buffer_length ) +{ + (void) buffer_length; + + netcode_assert( challenge_token ); + netcode_assert( buffer ); + netcode_assert( buffer_length >= NETCODE_CHALLENGE_TOKEN_BYTES ); + + memset( buffer, 0, NETCODE_CHALLENGE_TOKEN_BYTES ); + + uint8_t * start = buffer; + + (void) start; + + netcode_write_uint64( &buffer, challenge_token->client_id ); + + netcode_write_bytes( &buffer, challenge_token->user_data, NETCODE_USER_DATA_BYTES ); + + netcode_assert( buffer - start <= NETCODE_CHALLENGE_TOKEN_BYTES - NETCODE_MAC_BYTES ); +} + +int netcode_encrypt_challenge_token( uint8_t * buffer, int buffer_length, uint64_t sequence, uint8_t * key ) +{ + netcode_assert( buffer ); + netcode_assert( buffer_length >= NETCODE_CHALLENGE_TOKEN_BYTES ); + netcode_assert( key ); + + (void) buffer_length; + + uint8_t nonce[12]; + { + uint8_t * p = nonce; + netcode_write_uint32( &p, 0 ); + netcode_write_uint64( &p, sequence ); + } + + return netcode_encrypt_aead( buffer, NETCODE_CHALLENGE_TOKEN_BYTES - NETCODE_MAC_BYTES, NULL, 0, nonce, key ); +} + +int netcode_decrypt_challenge_token( uint8_t * buffer, int buffer_length, uint64_t sequence, uint8_t * key ) +{ + netcode_assert( buffer ); + netcode_assert( buffer_length >= NETCODE_CHALLENGE_TOKEN_BYTES ); + netcode_assert( key ); + + (void) buffer_length; + + uint8_t nonce[12]; + { + uint8_t * p = nonce; + netcode_write_uint32( &p, 0 ); + netcode_write_uint64( &p, sequence ); + } + + return netcode_decrypt_aead( buffer, NETCODE_CHALLENGE_TOKEN_BYTES, NULL, 0, nonce, key ); +} + +int netcode_read_challenge_token( uint8_t * buffer, int buffer_length, struct netcode_challenge_token_t * challenge_token ) +{ + netcode_assert( buffer ); + netcode_assert( challenge_token ); + + if ( buffer_length < NETCODE_CHALLENGE_TOKEN_BYTES ) + return NETCODE_ERROR; + + uint8_t * start = buffer; + + (void) start; + + challenge_token->client_id = netcode_read_uint64( &buffer ); + + netcode_read_bytes( &buffer, challenge_token->user_data, NETCODE_USER_DATA_BYTES ); + + netcode_assert( buffer - start == 8 + NETCODE_USER_DATA_BYTES ); + + return NETCODE_OK; +} + +// ---------------------------------------------------------------- + +#define NETCODE_CONNECTION_REQUEST_PACKET 0 +#define NETCODE_CONNECTION_DENIED_PACKET 1 +#define NETCODE_CONNECTION_CHALLENGE_PACKET 2 +#define NETCODE_CONNECTION_RESPONSE_PACKET 3 +#define NETCODE_CONNECTION_KEEP_ALIVE_PACKET 4 +#define NETCODE_CONNECTION_PAYLOAD_PACKET 5 +#define NETCODE_CONNECTION_DISCONNECT_PACKET 6 +#define NETCODE_CONNECTION_NUM_PACKETS 7 + +struct netcode_connection_request_packet_t +{ + uint8_t packet_type; + uint8_t version_info[NETCODE_VERSION_INFO_BYTES]; + uint64_t protocol_id; + uint64_t connect_token_expire_timestamp; + uint8_t connect_token_nonce[NETCODE_CONNECT_TOKEN_NONCE_BYTES]; + uint8_t connect_token_data[NETCODE_CONNECT_TOKEN_PRIVATE_BYTES]; +}; + +struct netcode_connection_denied_packet_t +{ + uint8_t packet_type; +}; + +struct netcode_connection_challenge_packet_t +{ + uint8_t packet_type; + uint64_t challenge_token_sequence; + uint8_t challenge_token_data[NETCODE_CHALLENGE_TOKEN_BYTES]; +}; + +struct netcode_connection_response_packet_t +{ + uint8_t packet_type; + uint64_t challenge_token_sequence; + uint8_t challenge_token_data[NETCODE_CHALLENGE_TOKEN_BYTES]; +}; + +struct netcode_connection_keep_alive_packet_t +{ + uint8_t packet_type; + int client_index; + int max_clients; +}; + +struct netcode_connection_payload_packet_t +{ + uint8_t packet_type; + uint32_t payload_bytes; + uint8_t payload_data[1]; +}; + +struct netcode_connection_disconnect_packet_t +{ + uint8_t packet_type; +}; + +struct netcode_connection_payload_packet_t * netcode_create_payload_packet( int payload_bytes, void * allocator_context, void* (*allocate_function)(void*,uint64_t) ) +{ + netcode_assert( payload_bytes >= 0 ); + netcode_assert( payload_bytes <= NETCODE_MAX_PAYLOAD_BYTES ); + + if ( allocate_function == NULL ) + { + allocate_function = netcode_default_allocate_function; + } + + struct netcode_connection_payload_packet_t * packet = (struct netcode_connection_payload_packet_t*) + allocate_function( allocator_context, sizeof( struct netcode_connection_payload_packet_t ) + payload_bytes ); + + if ( !packet ) + return NULL; + + packet->packet_type = NETCODE_CONNECTION_PAYLOAD_PACKET; + packet->payload_bytes = payload_bytes; + + return packet; +} + +struct netcode_context_t +{ + uint8_t write_packet_key[NETCODE_KEY_BYTES]; + uint8_t read_packet_key[NETCODE_KEY_BYTES]; +}; + +int netcode_sequence_number_bytes_required( uint64_t sequence ) +{ + int i; + uint64_t mask = 0xFF00000000000000UL; + for ( i = 0; i < 7; ++i ) + { + if ( sequence & mask ) + break; + mask >>= 8; + } + return 8 - i; +} + +int netcode_write_packet( void * packet, uint8_t * buffer, int buffer_length, uint64_t sequence, uint8_t * write_packet_key, uint64_t protocol_id ) +{ + netcode_assert( packet ); + netcode_assert( buffer ); + netcode_assert( write_packet_key ); + + (void) buffer_length; + + uint8_t packet_type = ((uint8_t*)packet)[0]; + + if ( packet_type == NETCODE_CONNECTION_REQUEST_PACKET ) + { + // connection request packet: first byte is zero + + netcode_assert( buffer_length >= 1 + 13 + 8 + 8 + NETCODE_CONNECT_TOKEN_NONCE_BYTES + NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + struct netcode_connection_request_packet_t * p = (struct netcode_connection_request_packet_t*) packet; + + uint8_t * start = buffer; + + netcode_write_uint8( &buffer, NETCODE_CONNECTION_REQUEST_PACKET ); + netcode_write_bytes( &buffer, p->version_info, NETCODE_VERSION_INFO_BYTES ); + netcode_write_uint64( &buffer, p->protocol_id ); + netcode_write_uint64( &buffer, p->connect_token_expire_timestamp ); + netcode_write_bytes( &buffer, p->connect_token_nonce, NETCODE_CONNECT_TOKEN_NONCE_BYTES ); + netcode_write_bytes( &buffer, p->connect_token_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + netcode_assert( buffer - start == 1 + 13 + 8 + 8 + NETCODE_CONNECT_TOKEN_NONCE_BYTES + NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + return (int) ( buffer - start ); + } + else + { + // *** encrypted packets *** + + // write the prefix byte (this is a combination of the packet type and number of sequence bytes) + + uint8_t * start = buffer; + + uint8_t sequence_bytes = (uint8_t) netcode_sequence_number_bytes_required( sequence ); + + netcode_assert( sequence_bytes >= 1 ); + netcode_assert( sequence_bytes <= 8 ); + + netcode_assert( packet_type <= 0xF ); + + uint8_t prefix_byte = packet_type | ( sequence_bytes << 4 ); + + netcode_write_uint8( &buffer, prefix_byte ); + + // write the variable length sequence number [1,8] bytes. + + uint64_t sequence_temp = sequence; + + int i; + for ( i = 0; i < sequence_bytes; ++i ) + { + netcode_write_uint8( &buffer, (uint8_t) ( sequence_temp & 0xFF ) ); + sequence_temp >>= 8; + } + + // write packet data according to type. this data will be encrypted. + + uint8_t * encrypted_start = buffer; + + switch ( packet_type ) + { + case NETCODE_CONNECTION_DENIED_PACKET: + { + // ... + } + break; + + case NETCODE_CONNECTION_CHALLENGE_PACKET: + { + struct netcode_connection_challenge_packet_t * p = (struct netcode_connection_challenge_packet_t*) packet; + netcode_write_uint64( &buffer, p->challenge_token_sequence ); + netcode_write_bytes( &buffer, p->challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES ); + } + break; + + case NETCODE_CONNECTION_RESPONSE_PACKET: + { + struct netcode_connection_response_packet_t * p = (struct netcode_connection_response_packet_t*) packet; + netcode_write_uint64( &buffer, p->challenge_token_sequence ); + netcode_write_bytes( &buffer, p->challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES ); + } + break; + + case NETCODE_CONNECTION_KEEP_ALIVE_PACKET: + { + struct netcode_connection_keep_alive_packet_t * p = (struct netcode_connection_keep_alive_packet_t*) packet; + netcode_write_uint32( &buffer, p->client_index ); + netcode_write_uint32( &buffer, p->max_clients ); + } + break; + + case NETCODE_CONNECTION_PAYLOAD_PACKET: + { + struct netcode_connection_payload_packet_t * p = (struct netcode_connection_payload_packet_t*) packet; + + netcode_assert( p->payload_bytes <= NETCODE_MAX_PAYLOAD_BYTES ); + + netcode_write_bytes( &buffer, p->payload_data, p->payload_bytes ); + } + break; + + case NETCODE_CONNECTION_DISCONNECT_PACKET: + { + // ... + } + break; + + default: + netcode_assert( 0 ); + } + + netcode_assert( buffer - start <= buffer_length - NETCODE_MAC_BYTES ); + + uint8_t * encrypted_finish = buffer; + + // encrypt the per-packet packet written with the prefix byte, protocol id and version as the associated data. this must match to decrypt. + + uint8_t additional_data[NETCODE_VERSION_INFO_BYTES+8+1]; + { + uint8_t * p = additional_data; + netcode_write_bytes( &p, NETCODE_VERSION_INFO, NETCODE_VERSION_INFO_BYTES ); + netcode_write_uint64( &p, protocol_id ); + netcode_write_uint8( &p, prefix_byte ); + } + + uint8_t nonce[12]; + { + uint8_t * p = nonce; + netcode_write_uint32( &p, 0 ); + netcode_write_uint64( &p, sequence ); + } + + if ( netcode_encrypt_aead( encrypted_start, + encrypted_finish - encrypted_start, + additional_data, sizeof( additional_data ), + nonce, write_packet_key ) != NETCODE_OK ) + { + return NETCODE_ERROR; + } + + buffer += NETCODE_MAC_BYTES; + + netcode_assert( buffer - start <= buffer_length ); + + return (int) ( buffer - start ); + } +} + +struct netcode_replay_protection_t +{ + uint64_t most_recent_sequence; + uint64_t received_packet[NETCODE_REPLAY_PROTECTION_BUFFER_SIZE]; +}; + +void netcode_replay_protection_reset( struct netcode_replay_protection_t * replay_protection ) +{ + netcode_assert( replay_protection ); + replay_protection->most_recent_sequence = 0; + memset( replay_protection->received_packet, 0xFF, sizeof( replay_protection->received_packet ) ); +} + +int netcode_replay_protection_already_received( struct netcode_replay_protection_t * replay_protection, uint64_t sequence ) +{ + netcode_assert( replay_protection ); + + if ( sequence + NETCODE_REPLAY_PROTECTION_BUFFER_SIZE <= replay_protection->most_recent_sequence ) + return 1; + + int index = (int) ( sequence % NETCODE_REPLAY_PROTECTION_BUFFER_SIZE ); + + if ( replay_protection->received_packet[index] == UINT64_MAX ) + return 0; + + if ( replay_protection->received_packet[index] >= sequence ) + return 1; + + return 0; +} + +void netcode_replay_protection_advance_sequence( struct netcode_replay_protection_t * replay_protection, uint64_t sequence ) +{ + netcode_assert( replay_protection ); + + if ( sequence > replay_protection->most_recent_sequence ) + replay_protection->most_recent_sequence = sequence; + + int index = (int) ( sequence % NETCODE_REPLAY_PROTECTION_BUFFER_SIZE ); + + replay_protection->received_packet[index] = sequence; +} + +void * netcode_read_packet( uint8_t * buffer, + int buffer_length, + uint64_t * sequence, + uint8_t * read_packet_key, + uint64_t protocol_id, + uint64_t current_timestamp, + uint8_t * private_key, + uint8_t * allowed_packets, + struct netcode_replay_protection_t * replay_protection, + void * allocator_context, + void* (*allocate_function)(void*,uint64_t) ) +{ + netcode_assert( sequence ); + netcode_assert( allowed_packets ); + + *sequence = 0; + + if ( allocate_function == NULL ) + { + allocate_function = netcode_default_allocate_function; + } + + if ( buffer_length < 1 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored packet. buffer length is less than 1\n" ); + return NULL; + } + + uint8_t * start = buffer; + + uint8_t prefix_byte = netcode_read_uint8( &buffer ); + + if ( prefix_byte == NETCODE_CONNECTION_REQUEST_PACKET ) + { + // connection request packet: first byte is zero + + if ( !allowed_packets[NETCODE_CONNECTION_REQUEST_PACKET] ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection request packet. packet type is not allowed\n" ); + return NULL; + } + + if ( buffer_length != 1 + NETCODE_VERSION_INFO_BYTES + 8 + 8 + NETCODE_CONNECT_TOKEN_NONCE_BYTES + NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection request packet. bad packet length (expected %d, got %d)\n", 1 + NETCODE_VERSION_INFO_BYTES + 8 + 8 + 8 + NETCODE_CONNECT_TOKEN_PRIVATE_BYTES, buffer_length ); + return NULL; + } + + if ( !private_key ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection request packet. no private key\n" ); + return NULL; + } + + uint8_t version_info[NETCODE_VERSION_INFO_BYTES]; + netcode_read_bytes( &buffer, version_info, NETCODE_VERSION_INFO_BYTES ); + if ( version_info[0] != 'N' || + version_info[1] != 'E' || + version_info[2] != 'T' || + version_info[3] != 'C' || + version_info[4] != 'O' || + version_info[5] != 'D' || + version_info[6] != 'E' || + version_info[7] != ' ' || + version_info[8] != '1' || + version_info[9] != '.' || + version_info[10] != '0' || + version_info[11] != '2' || + version_info[12] != '\0' ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection request packet. bad version info\n" ); + return NULL; + } + + uint64_t packet_protocol_id = netcode_read_uint64( &buffer ); + if ( packet_protocol_id != protocol_id ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection request packet. wrong protocol id. expected %.16" PRIx64 ", got %.16" PRIx64 "\n", + protocol_id, packet_protocol_id ); + return NULL; + } + + uint64_t packet_connect_token_expire_timestamp = netcode_read_uint64( &buffer ); + if ( packet_connect_token_expire_timestamp <= current_timestamp ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection request packet. connect token expired\n" ); + return NULL; + } + + uint8_t packet_connect_token_nonce[NETCODE_CONNECT_TOKEN_NONCE_BYTES]; + netcode_read_bytes(&buffer, packet_connect_token_nonce, sizeof(packet_connect_token_nonce)); + + netcode_assert( buffer - start == 1 + NETCODE_VERSION_INFO_BYTES + 8 + 8 + NETCODE_CONNECT_TOKEN_NONCE_BYTES ); + + if ( netcode_decrypt_connect_token_private( buffer, + NETCODE_CONNECT_TOKEN_PRIVATE_BYTES, + version_info, + protocol_id, + packet_connect_token_expire_timestamp, + packet_connect_token_nonce, + private_key ) != NETCODE_OK ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection request packet. connect token failed to decrypt\n" ); + return NULL; + } + + struct netcode_connection_request_packet_t * packet = (struct netcode_connection_request_packet_t*) + allocate_function( allocator_context, sizeof( struct netcode_connection_request_packet_t ) ); + + if ( !packet ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection request packet. failed to allocate packet\n" ); + return NULL; + } + + packet->packet_type = NETCODE_CONNECTION_REQUEST_PACKET; + memcpy( packet->version_info, version_info, NETCODE_VERSION_INFO_BYTES ); + packet->protocol_id = packet_protocol_id; + packet->connect_token_expire_timestamp = packet_connect_token_expire_timestamp; + memcpy( packet->connect_token_nonce, packet_connect_token_nonce, NETCODE_CONNECT_TOKEN_NONCE_BYTES ); + netcode_read_bytes( &buffer, packet->connect_token_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + netcode_assert( buffer - start == 1 + NETCODE_VERSION_INFO_BYTES + 8 + 8 + NETCODE_CONNECT_TOKEN_NONCE_BYTES + NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + return packet; + } + else + { + // *** encrypted packets *** + + if ( !read_packet_key ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored encrypted packet. no read packet key for this address\n" ); + return NULL; + } + + if ( buffer_length < 1 + 1 + NETCODE_MAC_BYTES ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored encrypted packet. packet is too small to be valid (%d bytes)\n", buffer_length ); + return NULL; + } + + // extract the packet type and number of sequence bytes from the prefix byte + + int packet_type = prefix_byte & 0xF; + + if ( packet_type >= NETCODE_CONNECTION_NUM_PACKETS ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored encrypted packet. packet type %d is invalid\n", packet_type ); + return NULL; + } + + if ( !allowed_packets[packet_type] ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored encrypted packet. packet type %d is not allowed\n", packet_type ); + return NULL; + } + + int sequence_bytes = prefix_byte >> 4; + + if ( sequence_bytes < 1 || sequence_bytes > 8 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored encrypted packet. sequence bytes %d is out of range [1,8]\n", sequence_bytes ); + return NULL; + } + + if ( buffer_length < 1 + sequence_bytes + NETCODE_MAC_BYTES ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored encrypted packet. buffer is too small for sequence bytes + encryption mac\n" ); + return NULL; + } + + // read variable length sequence number [1,8] + + int i; + for ( i = 0; i < sequence_bytes; ++i ) + { + uint8_t value = netcode_read_uint8( &buffer ); + (*sequence) |= ( uint64_t) ( value ) << ( 8 * i ); + } + + // ignore the packet if it has already been received + + if ( replay_protection && packet_type >= NETCODE_CONNECTION_KEEP_ALIVE_PACKET ) + { + if ( netcode_replay_protection_already_received( replay_protection, *sequence ) ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored packet. sequence %.16" PRIx64 " already received (replay protection)\n", *sequence ); + return NULL; + } + } + + // decrypt the per-packet type data + + uint8_t additional_data[NETCODE_VERSION_INFO_BYTES+8+1]; + { + uint8_t * p = additional_data; + netcode_write_bytes( &p, NETCODE_VERSION_INFO, NETCODE_VERSION_INFO_BYTES ); + netcode_write_uint64( &p, protocol_id ); + netcode_write_uint8( &p, prefix_byte ); + } + + uint8_t nonce[12]; + { + uint8_t * p = nonce; + netcode_write_uint32( &p, 0 ); + netcode_write_uint64( &p, *sequence ); + } + + int encrypted_bytes = (int) ( buffer_length - ( buffer - start ) ); + + if ( encrypted_bytes < NETCODE_MAC_BYTES ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored encrypted packet. encrypted payload is too small\n" ); + return NULL; + } + + if ( netcode_decrypt_aead( buffer, encrypted_bytes, additional_data, sizeof( additional_data ), nonce, read_packet_key ) != NETCODE_OK ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored encrypted packet. failed to decrypt\n" ); + return NULL; + } + + int decrypted_bytes = encrypted_bytes - NETCODE_MAC_BYTES; + + // update the latest replay protection sequence # + + if ( replay_protection && packet_type >= NETCODE_CONNECTION_KEEP_ALIVE_PACKET ) + { + netcode_replay_protection_advance_sequence( replay_protection, *sequence ); + } + + // process the per-packet type data that was just decrypted + + switch ( packet_type ) + { + case NETCODE_CONNECTION_DENIED_PACKET: + { + if ( decrypted_bytes != 0 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection denied packet. decrypted packet data is wrong size\n" ); + return NULL; + } + + struct netcode_connection_denied_packet_t * packet = (struct netcode_connection_denied_packet_t*) + allocate_function( allocator_context, sizeof( struct netcode_connection_denied_packet_t ) ); + + if ( !packet ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection denied packet. could not allocate packet struct\n" ); + return NULL; + } + + packet->packet_type = NETCODE_CONNECTION_DENIED_PACKET; + + return packet; + } + break; + + case NETCODE_CONNECTION_CHALLENGE_PACKET: + { + if ( decrypted_bytes != 8 + NETCODE_CHALLENGE_TOKEN_BYTES ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection challenge packet. decrypted packet data is wrong size\n" ); + return NULL; + } + + struct netcode_connection_challenge_packet_t * packet = (struct netcode_connection_challenge_packet_t*) + allocate_function( allocator_context, sizeof( struct netcode_connection_challenge_packet_t ) ); + + if ( !packet ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection challenge packet. could not allocate packet struct\n" ); + return NULL; + } + + packet->packet_type = NETCODE_CONNECTION_CHALLENGE_PACKET; + packet->challenge_token_sequence = netcode_read_uint64( &buffer ); + netcode_read_bytes( &buffer, packet->challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES ); + + return packet; + } + break; + + case NETCODE_CONNECTION_RESPONSE_PACKET: + { + if ( decrypted_bytes != 8 + NETCODE_CHALLENGE_TOKEN_BYTES ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection response packet. decrypted packet data is wrong size\n" ); + return NULL; + } + + struct netcode_connection_response_packet_t * packet = (struct netcode_connection_response_packet_t*) + allocate_function( allocator_context, sizeof( struct netcode_connection_response_packet_t ) ); + + if ( !packet ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection response packet. could not allocate packet struct\n" ); + return NULL; + } + + packet->packet_type = NETCODE_CONNECTION_RESPONSE_PACKET; + packet->challenge_token_sequence = netcode_read_uint64( &buffer ); + netcode_read_bytes( &buffer, packet->challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES ); + + return packet; + } + break; + + case NETCODE_CONNECTION_KEEP_ALIVE_PACKET: + { + if ( decrypted_bytes != 8 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection keep alive packet. decrypted packet data is wrong size\n" ); + return NULL; + } + + struct netcode_connection_keep_alive_packet_t * packet = (struct netcode_connection_keep_alive_packet_t*) + allocate_function( allocator_context, sizeof( struct netcode_connection_keep_alive_packet_t ) ); + + if ( !packet ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection keep alive packet. could not allocate packet struct\n" ); + return NULL; + } + + packet->packet_type = NETCODE_CONNECTION_KEEP_ALIVE_PACKET; + packet->client_index = netcode_read_uint32( &buffer ); + packet->max_clients = netcode_read_uint32( &buffer ); + + return packet; + } + break; + + case NETCODE_CONNECTION_PAYLOAD_PACKET: + { + if ( decrypted_bytes < 1 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection payload packet. payload is too small\n" ); + return NULL; + } + + if ( decrypted_bytes > NETCODE_MAX_PAYLOAD_BYTES ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection payload packet. payload is too large\n" ); + return NULL; + } + + struct netcode_connection_payload_packet_t * packet = netcode_create_payload_packet( decrypted_bytes, allocator_context, allocate_function ); + + if ( !packet ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection payload packet. could not allocate packet struct\n" ); + return NULL; + } + + memcpy( packet->payload_data, buffer, decrypted_bytes ); + + return packet; + } + break; + + case NETCODE_CONNECTION_DISCONNECT_PACKET: + { + if ( decrypted_bytes != 0 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection disconnect packet. decrypted packet data is wrong size\n" ); + return NULL; + } + + struct netcode_connection_disconnect_packet_t * packet = (struct netcode_connection_disconnect_packet_t*) + allocate_function( allocator_context, sizeof( struct netcode_connection_disconnect_packet_t ) ); + + if ( !packet ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "ignored connection disconnect packet. could not allocate packet struct\n" ); + return NULL; + } + + packet->packet_type = NETCODE_CONNECTION_DISCONNECT_PACKET; + + return packet; + } + break; + + default: + return NULL; + } + } +} + +// ---------------------------------------------------------------- + +struct netcode_connect_token_t +{ + uint8_t version_info[NETCODE_VERSION_INFO_BYTES]; + uint64_t protocol_id; + uint64_t create_timestamp; + uint64_t expire_timestamp; + uint8_t nonce[NETCODE_CONNECT_TOKEN_NONCE_BYTES]; + uint8_t private_data[NETCODE_CONNECT_TOKEN_PRIVATE_BYTES]; + int timeout_seconds; + int num_server_addresses; + struct netcode_address_t server_addresses[NETCODE_MAX_SERVERS_PER_CONNECT]; + uint8_t client_to_server_key[NETCODE_KEY_BYTES]; + uint8_t server_to_client_key[NETCODE_KEY_BYTES]; +}; + +void netcode_write_connect_token( struct netcode_connect_token_t * connect_token, uint8_t * buffer, int buffer_length ) +{ + netcode_assert( connect_token ); + netcode_assert( buffer ); + netcode_assert( buffer_length >= NETCODE_CONNECT_TOKEN_BYTES ); + + uint8_t * start = buffer; + + (void) start; + (void) buffer_length; + + netcode_write_bytes( &buffer, connect_token->version_info, NETCODE_VERSION_INFO_BYTES ); + + netcode_write_uint64( &buffer, connect_token->protocol_id ); + + netcode_write_uint64( &buffer, connect_token->create_timestamp ); + + netcode_write_uint64( &buffer, connect_token->expire_timestamp ); + + netcode_write_bytes( &buffer, connect_token->nonce, NETCODE_CONNECT_TOKEN_NONCE_BYTES ); + + netcode_write_bytes( &buffer, connect_token->private_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + int i,j; + + netcode_write_uint32( &buffer, connect_token->timeout_seconds ); + + netcode_write_uint32( &buffer, connect_token->num_server_addresses ); + + for ( i = 0; i < connect_token->num_server_addresses; ++i ) + { + if ( connect_token->server_addresses[i].type == NETCODE_ADDRESS_IPV4 ) + { + netcode_write_uint8( &buffer, NETCODE_ADDRESS_IPV4 ); + for ( j = 0; j < 4; ++j ) + { + netcode_write_uint8( &buffer, connect_token->server_addresses[i].data.ipv4[j] ); + } + netcode_write_uint16( &buffer, connect_token->server_addresses[i].port ); + } + else if ( connect_token->server_addresses[i].type == NETCODE_ADDRESS_IPV6 ) + { + netcode_write_uint8( &buffer, NETCODE_ADDRESS_IPV6 ); + for ( j = 0; j < 8; ++j ) + { + netcode_write_uint16( &buffer, connect_token->server_addresses[i].data.ipv6[j] ); + } + netcode_write_uint16( &buffer, connect_token->server_addresses[i].port ); + } + else + { + netcode_assert( 0 ); + } + } + + netcode_write_bytes( &buffer, connect_token->client_to_server_key, NETCODE_KEY_BYTES ); + + netcode_write_bytes( &buffer, connect_token->server_to_client_key, NETCODE_KEY_BYTES ); + + netcode_assert( buffer - start <= NETCODE_CONNECT_TOKEN_BYTES ); + + memset( buffer, 0, NETCODE_CONNECT_TOKEN_BYTES - ( buffer - start ) ); +} + +int netcode_read_connect_token( uint8_t * buffer, int buffer_length, struct netcode_connect_token_t * connect_token ) +{ + netcode_assert( buffer ); + netcode_assert( connect_token ); + + if ( buffer_length != NETCODE_CONNECT_TOKEN_BYTES ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: read connect data has bad buffer length (%d)\n", buffer_length ); + return NETCODE_ERROR; + } + + netcode_read_bytes( &buffer, connect_token->version_info, NETCODE_VERSION_INFO_BYTES ); + if ( connect_token->version_info[0] != 'N' || + connect_token->version_info[1] != 'E' || + connect_token->version_info[2] != 'T' || + connect_token->version_info[3] != 'C' || + connect_token->version_info[4] != 'O' || + connect_token->version_info[5] != 'D' || + connect_token->version_info[6] != 'E' || + connect_token->version_info[7] != ' ' || + connect_token->version_info[8] != '1' || + connect_token->version_info[9] != '.' || + connect_token->version_info[10] != '0' || + connect_token->version_info[11] != '2' || + connect_token->version_info[12] != '\0' ) + { + connect_token->version_info[12] = '\0'; + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: read connect data has bad version info (got %s, expected %s)\n", connect_token->version_info, NETCODE_VERSION_INFO ); + return NETCODE_ERROR; + } + + connect_token->protocol_id = netcode_read_uint64( &buffer ); + + connect_token->create_timestamp = netcode_read_uint64( &buffer ); + + connect_token->expire_timestamp = netcode_read_uint64( &buffer ); + + if ( connect_token->create_timestamp > connect_token->expire_timestamp ) + return NETCODE_ERROR; + + netcode_read_bytes( &buffer, connect_token->nonce, NETCODE_CONNECT_TOKEN_NONCE_BYTES ); + + netcode_read_bytes( &buffer, connect_token->private_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + connect_token->timeout_seconds = (int) netcode_read_uint32( &buffer ); + + connect_token->num_server_addresses = netcode_read_uint32( &buffer ); + + if ( connect_token->num_server_addresses <= 0 || connect_token->num_server_addresses > NETCODE_MAX_SERVERS_PER_CONNECT ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: read connect data has bad number of server addresses (%d)\n", connect_token->num_server_addresses ); + return NETCODE_ERROR; + } + + int i,j; + + for ( i = 0; i < connect_token->num_server_addresses; ++i ) + { + connect_token->server_addresses[i].type = netcode_read_uint8( &buffer ); + + if ( connect_token->server_addresses[i].type == NETCODE_ADDRESS_IPV4 ) + { + for ( j = 0; j < 4; ++j ) + { + connect_token->server_addresses[i].data.ipv4[j] = netcode_read_uint8( &buffer ); + } + connect_token->server_addresses[i].port = netcode_read_uint16( &buffer ); + } + else if ( connect_token->server_addresses[i].type == NETCODE_ADDRESS_IPV6 ) + { + for ( j = 0; j < 8; ++j ) + { + connect_token->server_addresses[i].data.ipv6[j] = netcode_read_uint16( &buffer ); + } + connect_token->server_addresses[i].port = netcode_read_uint16( &buffer ); + } + else + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: read connect data has bad address type (%d)\n", connect_token->server_addresses[i].type ); + return NETCODE_ERROR; + } + } + + netcode_read_bytes( &buffer, connect_token->client_to_server_key, NETCODE_KEY_BYTES ); + + netcode_read_bytes( &buffer, connect_token->server_to_client_key, NETCODE_KEY_BYTES ); + + return NETCODE_OK; +} + +// ---------------------------------------------------------------- + +struct netcode_packet_queue_t +{ + void * allocator_context; + void * (*allocate_function)(void*,uint64_t); + void (*free_function)(void*,void*); + int num_packets; + int start_index; + void * packet_data[NETCODE_PACKET_QUEUE_SIZE]; + uint64_t packet_sequence[NETCODE_PACKET_QUEUE_SIZE]; +}; + +void netcode_packet_queue_init( struct netcode_packet_queue_t * queue, + void * allocator_context, + void * (*allocate_function)(void*,uint64_t), + void (*free_function)(void*,void*) ) +{ + if ( allocate_function == NULL ) + { + allocate_function = netcode_default_allocate_function; + } + + if ( free_function == NULL ) + { + free_function = netcode_default_free_function; + } + + netcode_assert( queue ); + + queue->allocator_context = allocator_context; + queue->allocate_function = allocate_function; + queue->free_function = free_function; + queue->num_packets = 0; + queue->start_index = 0; + memset( queue->packet_data, 0, sizeof( queue->packet_data ) ); + memset( queue->packet_sequence, 0, sizeof( queue->packet_sequence ) ); +} + +void netcode_packet_queue_clear( struct netcode_packet_queue_t * queue ) +{ + int i; + for ( i = 0; i < queue->num_packets; ++i ) + { + queue->free_function( queue->allocator_context, queue->packet_data[i] ); + } + queue->num_packets = 0; + queue->start_index = 0; + memset( queue->packet_data, 0, sizeof( queue->packet_data ) ); + memset( queue->packet_sequence, 0, sizeof( queue->packet_sequence ) ); +} + +int netcode_packet_queue_push( struct netcode_packet_queue_t * queue, void * packet_data, uint64_t packet_sequence ) +{ + netcode_assert( queue ); + netcode_assert( packet_data ); + if ( queue->num_packets == NETCODE_PACKET_QUEUE_SIZE ) + { + queue->free_function( queue->allocator_context, packet_data ); + return 0; + } + int index = ( queue->start_index + queue->num_packets ) % NETCODE_PACKET_QUEUE_SIZE; + queue->packet_data[index] = packet_data; + queue->packet_sequence[index] = packet_sequence; + queue->num_packets++; + return 1; +} + +void * netcode_packet_queue_pop( struct netcode_packet_queue_t * queue, uint64_t * packet_sequence ) +{ + if ( queue->num_packets == 0 ) + return NULL; + void * packet = queue->packet_data[queue->start_index]; + if ( packet_sequence ) + *packet_sequence = queue->packet_sequence[queue->start_index]; + queue->start_index = ( queue->start_index + 1 ) % NETCODE_PACKET_QUEUE_SIZE; + queue->num_packets--; + return packet; +} + +// ---------------------------------------------------------------- + +#define NETCODE_NETWORK_SIMULATOR_NUM_PACKET_ENTRIES ( NETCODE_MAX_CLIENTS * 256 ) +#define NETCODE_NETWORK_SIMULATOR_NUM_PENDING_RECEIVE_PACKETS ( NETCODE_MAX_CLIENTS * 64 ) + +struct netcode_network_simulator_packet_entry_t +{ + struct netcode_address_t from; + struct netcode_address_t to; + double delivery_time; + uint8_t * packet_data; + int packet_bytes; +}; + +struct netcode_network_simulator_t +{ + void * allocator_context; + void * (*allocate_function)(void*,uint64_t); + void (*free_function)(void*,void*); + float latency_milliseconds; + float jitter_milliseconds; + float packet_loss_percent; + float duplicate_packet_percent; + double time; + int current_index; + int num_pending_receive_packets; + struct netcode_network_simulator_packet_entry_t packet_entries[NETCODE_NETWORK_SIMULATOR_NUM_PACKET_ENTRIES]; + struct netcode_network_simulator_packet_entry_t pending_receive_packets[NETCODE_NETWORK_SIMULATOR_NUM_PENDING_RECEIVE_PACKETS]; +}; + +struct netcode_network_simulator_t * netcode_network_simulator_create( void * allocator_context, + void * (*allocate_function)(void*,uint64_t), + void (*free_function)(void*,void*) ) +{ + if ( allocate_function == NULL ) + { + allocate_function = netcode_default_allocate_function; + } + + if ( free_function == NULL ) + { + free_function = netcode_default_free_function; + } + + struct netcode_network_simulator_t * network_simulator = (struct netcode_network_simulator_t*) + allocate_function( allocator_context, sizeof( struct netcode_network_simulator_t ) ); + + netcode_assert( network_simulator ); + + memset( network_simulator, 0, sizeof( struct netcode_network_simulator_t ) ); + + network_simulator->allocator_context = allocator_context; + network_simulator->allocate_function = allocate_function; + network_simulator->free_function = free_function; + + return network_simulator; +} + +void netcode_network_simulator_reset( struct netcode_network_simulator_t * network_simulator ) +{ + netcode_assert( network_simulator ); + + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "network simulator reset\n" ); + + int i; + for ( i = 0; i < NETCODE_NETWORK_SIMULATOR_NUM_PACKET_ENTRIES; ++i ) + { + network_simulator->free_function( network_simulator->allocator_context, network_simulator->packet_entries[i].packet_data ); + memset( &network_simulator->packet_entries[i], 0, sizeof( struct netcode_network_simulator_packet_entry_t ) ); + } + + for ( i = 0; i < network_simulator->num_pending_receive_packets; ++i ) + { + network_simulator->free_function( network_simulator->allocator_context, network_simulator->pending_receive_packets[i].packet_data ); + memset( &network_simulator->pending_receive_packets[i], 0, sizeof( struct netcode_network_simulator_packet_entry_t ) ); + } + + network_simulator->current_index = 0; + network_simulator->num_pending_receive_packets = 0; +} + +void netcode_network_simulator_destroy( struct netcode_network_simulator_t * network_simulator ) +{ + netcode_assert( network_simulator ); + netcode_network_simulator_reset( network_simulator ); + network_simulator->free_function( network_simulator->allocator_context, network_simulator ); +} + +float netcode_random_float( float a, float b ) +{ + netcode_assert( a < b ); + float random = ( (float) rand() ) / (float) RAND_MAX; + float diff = b - a; + float r = random * diff; + return a + r; +} + +void netcode_network_simulator_queue_packet( struct netcode_network_simulator_t * network_simulator, + struct netcode_address_t * from, + struct netcode_address_t * to, + uint8_t * packet_data, + int packet_bytes, + float delay ) +{ + network_simulator->packet_entries[network_simulator->current_index].from = *from; + network_simulator->packet_entries[network_simulator->current_index].to = *to; + network_simulator->packet_entries[network_simulator->current_index].packet_data = + (uint8_t*) network_simulator->allocate_function( network_simulator->allocator_context, packet_bytes ); + memcpy( network_simulator->packet_entries[network_simulator->current_index].packet_data, packet_data, packet_bytes ); + network_simulator->packet_entries[network_simulator->current_index].packet_bytes = packet_bytes; + network_simulator->packet_entries[network_simulator->current_index].delivery_time = network_simulator->time + delay; + network_simulator->current_index++; + network_simulator->current_index %= NETCODE_NETWORK_SIMULATOR_NUM_PACKET_ENTRIES; +} + +void netcode_network_simulator_send_packet( struct netcode_network_simulator_t * network_simulator, + struct netcode_address_t * from, + struct netcode_address_t * to, + uint8_t * packet_data, + int packet_bytes ) +{ + netcode_assert( network_simulator ); + netcode_assert( from ); + netcode_assert( from->type != 0 ); + netcode_assert( to ); + netcode_assert( to->type != 0 ); + netcode_assert( packet_data ); + netcode_assert( packet_bytes > 0 ); + netcode_assert( packet_bytes <= NETCODE_MAX_PACKET_BYTES ); + + if ( netcode_random_float( 0.0f, 100.0f ) <= network_simulator->packet_loss_percent ) + return; + + if ( network_simulator->packet_entries[network_simulator->current_index].packet_data ) + { + network_simulator->free_function( network_simulator->allocator_context, network_simulator->packet_entries[network_simulator->current_index].packet_data ); + network_simulator->packet_entries[network_simulator->current_index].packet_data = NULL; + } + + float delay = network_simulator->latency_milliseconds / 1000.0f; + + if ( network_simulator->jitter_milliseconds > 0.0 ) + delay += netcode_random_float( -network_simulator->jitter_milliseconds, +network_simulator->jitter_milliseconds ) / 1000.0f; + + netcode_network_simulator_queue_packet( network_simulator, from, to, packet_data, packet_bytes, delay ); + + if ( netcode_random_float( 0.0f, 100.0f ) <= network_simulator->duplicate_packet_percent ) + { + netcode_network_simulator_queue_packet( network_simulator, from, to, packet_data, packet_bytes, delay + netcode_random_float( 0, 1.0 ) ); + } +} + +int netcode_network_simulator_receive_packets( struct netcode_network_simulator_t * network_simulator, + struct netcode_address_t * to, + int max_packets, + uint8_t ** packet_data, + int * packet_bytes, + struct netcode_address_t * from ) +{ + netcode_assert( network_simulator ); + netcode_assert( max_packets >= 0 ); + netcode_assert( packet_data ); + netcode_assert( packet_bytes ); + netcode_assert( from ); + netcode_assert( to ); + + int num_packets = 0; + + int i; + for ( i = 0; i < network_simulator->num_pending_receive_packets; ++i ) + { + if ( num_packets == max_packets ) + break; + + if ( !network_simulator->pending_receive_packets[i].packet_data ) + continue; + + if ( !netcode_address_equal( &network_simulator->pending_receive_packets[i].to, to ) ) + continue; + + packet_data[num_packets] = network_simulator->pending_receive_packets[i].packet_data; + packet_bytes[num_packets] = network_simulator->pending_receive_packets[i].packet_bytes; + from[num_packets] = network_simulator->pending_receive_packets[i].from; + + network_simulator->pending_receive_packets[i].packet_data = NULL; + + num_packets++; + } + + netcode_assert( num_packets <= max_packets ); + + return num_packets; +} + +void netcode_network_simulator_update( struct netcode_network_simulator_t * network_simulator, double time ) +{ + netcode_assert( network_simulator ); + + network_simulator->time = time; + + // discard any pending receive packets that are still in the buffer + + int i; + for ( i = 0; i < network_simulator->num_pending_receive_packets; ++i ) + { + if ( network_simulator->pending_receive_packets[i].packet_data ) + { + network_simulator->free_function( network_simulator->allocator_context, network_simulator->pending_receive_packets[i].packet_data ); + network_simulator->pending_receive_packets[i].packet_data = NULL; + } + } + + network_simulator->num_pending_receive_packets = 0; + + // walk across packet entries and move any that are ready to be received into the pending receive buffer + + for ( i = 0; i < NETCODE_NETWORK_SIMULATOR_NUM_PACKET_ENTRIES; ++i ) + { + if ( !network_simulator->packet_entries[i].packet_data ) + continue; + + if ( network_simulator->num_pending_receive_packets == NETCODE_NETWORK_SIMULATOR_NUM_PENDING_RECEIVE_PACKETS ) + break; + + if ( network_simulator->packet_entries[i].packet_data && network_simulator->packet_entries[i].delivery_time <= time ) + { + network_simulator->pending_receive_packets[network_simulator->num_pending_receive_packets] = network_simulator->packet_entries[i]; + network_simulator->num_pending_receive_packets++; + network_simulator->packet_entries[i].packet_data = NULL; + } + } +} + +// ---------------------------------------------------------------- + +NETCODE_CONST char * netcode_client_state_name( int client_state ) +{ + switch ( client_state ) + { + case NETCODE_CLIENT_STATE_CONNECT_TOKEN_EXPIRED: return "connect token expired"; + case NETCODE_CLIENT_STATE_INVALID_CONNECT_TOKEN: return "invalid connect token"; + case NETCODE_CLIENT_STATE_CONNECTION_TIMED_OUT: return "connection timed out"; + case NETCODE_CLIENT_STATE_CONNECTION_REQUEST_TIMED_OUT: return "connection request timed out"; + case NETCODE_CLIENT_STATE_CONNECTION_RESPONSE_TIMED_OUT: return "connection response timed out"; + case NETCODE_CLIENT_STATE_CONNECTION_DENIED: return "connection denied"; + case NETCODE_CLIENT_STATE_DISCONNECTED: return "disconnected"; + case NETCODE_CLIENT_STATE_SENDING_CONNECTION_REQUEST: return "sending connection request"; + case NETCODE_CLIENT_STATE_SENDING_CONNECTION_RESPONSE: return "sending connection response"; + case NETCODE_CLIENT_STATE_CONNECTED: return "connected"; + default: + netcode_assert( 0 ); + return "???"; + } +} + +void netcode_default_client_config( struct netcode_client_config_t * config ) +{ + netcode_assert( config ); + config->allocator_context = NULL; + config->allocate_function = netcode_default_allocate_function; + config->free_function = netcode_default_free_function; + config->network_simulator = NULL; + config->callback_context = NULL; + config->state_change_callback = NULL; + config->send_loopback_packet_callback = NULL; + config->override_send_and_receive = 0; + config->send_packet_override = NULL; + config->receive_packet_override = NULL; +}; + +struct netcode_client_t +{ + struct netcode_client_config_t config; + int state; + double time; + double connect_start_time; + double last_packet_send_time; + double last_packet_receive_time; + int should_disconnect; + int should_disconnect_state; + uint64_t sequence; + int client_index; + int max_clients; + int server_address_index; + struct netcode_address_t address; + struct netcode_address_t server_address; + struct netcode_connect_token_t connect_token; + struct netcode_socket_holder_t socket_holder; + struct netcode_context_t context; + struct netcode_replay_protection_t replay_protection; + struct netcode_packet_queue_t packet_receive_queue; + uint64_t challenge_token_sequence; + uint8_t challenge_token_data[NETCODE_CHALLENGE_TOKEN_BYTES]; + uint8_t * receive_packet_data[NETCODE_CLIENT_MAX_RECEIVE_PACKETS]; + int receive_packet_bytes[NETCODE_CLIENT_MAX_RECEIVE_PACKETS]; + struct netcode_address_t receive_from[NETCODE_CLIENT_MAX_RECEIVE_PACKETS]; + int loopback; +}; + +int netcode_client_socket_create( struct netcode_socket_t * socket, + struct netcode_address_t * address, + int send_buffer_size, + int receive_buffer_size, + NETCODE_CONST struct netcode_client_config_t * config ) +{ + netcode_assert( socket ); + netcode_assert( address ); + netcode_assert( config ); + + if ( !config->network_simulator ) + { + if ( !config->override_send_and_receive ) + { + if ( netcode_socket_create( socket, address, send_buffer_size, receive_buffer_size ) != NETCODE_SOCKET_ERROR_NONE ) + { + return 0; + } + } + } + else + { + if ( address->port == 0 ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: must bind to a specific port when using network simulator\n" ); + return 0; + } + } + + return 1; +} + +struct netcode_client_t * netcode_client_create_overload( NETCODE_CONST char * address1_string, + NETCODE_CONST char * address2_string, + NETCODE_CONST struct netcode_client_config_t * config, + double time ) +{ + netcode_assert( config ); + netcode_assert( netcode.initialized ); + + struct netcode_address_t address1; + struct netcode_address_t address2; + + memset( &address1, 0, sizeof( address1 ) ); + memset( &address2, 0, sizeof( address2 ) ); + + if ( netcode_parse_address( address1_string, &address1 ) != NETCODE_OK ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to parse client address\n" ); + return NULL; + } + + if ( address2_string != NULL && netcode_parse_address( address2_string, &address2 ) != NETCODE_OK ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to parse client address2\n" ); + return NULL; + } + + + struct netcode_socket_t socket_ipv4; + struct netcode_socket_t socket_ipv6; + + memset( &socket_ipv4, 0, sizeof( socket_ipv4 ) ); + memset( &socket_ipv6, 0, sizeof( socket_ipv6 ) ); + + if ( address1.type == NETCODE_ADDRESS_IPV4 || address2.type == NETCODE_ADDRESS_IPV4 ) + { + if ( !netcode_client_socket_create( &socket_ipv4, address1.type == NETCODE_ADDRESS_IPV4 ? &address1 : &address2, NETCODE_CLIENT_SOCKET_SNDBUF_SIZE, NETCODE_CLIENT_SOCKET_RCVBUF_SIZE, config ) ) + { + return NULL; + } + } + + if ( address1.type == NETCODE_ADDRESS_IPV6 || address2.type == NETCODE_ADDRESS_IPV6 ) + { + if ( !netcode_client_socket_create( &socket_ipv6, address1.type == NETCODE_ADDRESS_IPV6 ? &address1 : &address2, NETCODE_CLIENT_SOCKET_SNDBUF_SIZE, NETCODE_CLIENT_SOCKET_RCVBUF_SIZE, config ) ) + { + return NULL; + } + } + + struct netcode_client_t * client = (struct netcode_client_t*) config->allocate_function( config->allocator_context, sizeof( struct netcode_client_t ) ); + + if ( !client ) + { + netcode_socket_destroy( &socket_ipv4 ); + netcode_socket_destroy( &socket_ipv6 ); + return NULL; + } + + struct netcode_address_t socket_address = address1.type == NETCODE_ADDRESS_IPV4 ? socket_ipv4.address : socket_ipv6.address; + + if ( !config->network_simulator ) + { + netcode_printf( NETCODE_LOG_LEVEL_INFO, "client started on port %d\n", socket_address.port ); + } + else + { + netcode_printf( NETCODE_LOG_LEVEL_INFO, "client started on port %d (network simulator)\n", socket_address.port ); + } + + client->config = *config; + client->socket_holder.ipv4 = socket_ipv4; + client->socket_holder.ipv6 = socket_ipv6; + client->address = config->network_simulator ? address1 : socket_address; + client->state = NETCODE_CLIENT_STATE_DISCONNECTED; + client->time = time; + client->connect_start_time = 0.0; + client->last_packet_send_time = -1000.0; + client->last_packet_receive_time = -1000.0; + client->should_disconnect = 0; + client->should_disconnect_state = NETCODE_CLIENT_STATE_DISCONNECTED; + client->sequence = 0; + client->client_index = 0; + client->max_clients = 0; + client->server_address_index = 0; + client->challenge_token_sequence = 0; + client->loopback = 0; + memset( &client->server_address, 0, sizeof( struct netcode_address_t ) ); + memset( &client->connect_token, 0, sizeof( struct netcode_connect_token_t ) ); + memset( &client->context, 0, sizeof( struct netcode_context_t ) ); + memset( client->challenge_token_data, 0, NETCODE_CHALLENGE_TOKEN_BYTES ); + + netcode_packet_queue_init( &client->packet_receive_queue, config->allocator_context, config->allocate_function, config->free_function ); + + netcode_replay_protection_reset( &client->replay_protection ); + + return client; +} + +struct netcode_client_t * netcode_client_create( NETCODE_CONST char * address, + NETCODE_CONST struct netcode_client_config_t * config, + double time ) +{ + return netcode_client_create_overload( address, NULL, config, time ); +} + +void netcode_client_destroy( struct netcode_client_t * client ) +{ + netcode_assert( client ); + if ( !client->loopback ) + netcode_client_disconnect( client ); + else + netcode_client_disconnect_loopback( client ); + netcode_socket_destroy( &client->socket_holder.ipv4 ); + netcode_socket_destroy( &client->socket_holder.ipv6 ); + netcode_packet_queue_clear( &client->packet_receive_queue ); + client->config.free_function( client->config.allocator_context, client ); +} + +void netcode_client_set_state( struct netcode_client_t * client, int client_state ) +{ + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client changed state from '%s' to '%s'\n", + netcode_client_state_name( client->state ), netcode_client_state_name( client_state ) ); + + if ( client->config.state_change_callback ) + { + client->config.state_change_callback( client->config.callback_context, client->state, client_state ); + } + + client->state = client_state; +} + +void netcode_client_reset_before_next_connect( struct netcode_client_t * client ) +{ + client->connect_start_time = client->time; + client->last_packet_send_time = client->time - 1.0f; + client->last_packet_receive_time = client->time; + client->should_disconnect = 0; + client->should_disconnect_state = NETCODE_CLIENT_STATE_DISCONNECTED; + client->challenge_token_sequence = 0; + + memset( client->challenge_token_data, 0, NETCODE_CHALLENGE_TOKEN_BYTES ); + + netcode_replay_protection_reset( &client->replay_protection ); +} + +void netcode_client_reset_connection_data( struct netcode_client_t * client, int client_state ) +{ + netcode_assert( client ); + + client->sequence = 0; + client->loopback = 0; + client->client_index = 0; + client->max_clients = 0; + client->connect_start_time = 0.0; + client->server_address_index = 0; + memset( &client->server_address, 0, sizeof( struct netcode_address_t ) ); + memset( &client->connect_token, 0, sizeof( struct netcode_connect_token_t ) ); + memset( &client->context, 0, sizeof( struct netcode_context_t ) ); + + netcode_client_set_state( client, client_state ); + + netcode_client_reset_before_next_connect( client ); + + while ( 1 ) + { + void * packet = netcode_packet_queue_pop( &client->packet_receive_queue, NULL ); + if ( !packet ) + break; + client->config.free_function( client->config.allocator_context, packet ); + } + + netcode_packet_queue_clear( &client->packet_receive_queue ); +} + +void netcode_client_disconnect_internal( struct netcode_client_t * client, int destination_state, int send_disconnect_packets ); + +void netcode_client_connect( struct netcode_client_t * client, uint8_t * connect_token ) +{ + netcode_assert( client ); + netcode_assert( connect_token ); + + netcode_client_disconnect( client ); + + if ( netcode_read_connect_token( connect_token, NETCODE_CONNECT_TOKEN_BYTES, &client->connect_token ) != NETCODE_OK ) + { + netcode_client_set_state( client, NETCODE_CLIENT_STATE_INVALID_CONNECT_TOKEN ); + return; + } + + client->server_address_index = 0; + client->server_address = client->connect_token.server_addresses[0]; + + char server_address_string[NETCODE_MAX_ADDRESS_STRING_LENGTH]; + + netcode_printf( NETCODE_LOG_LEVEL_INFO, "client connecting to server %s [%d/%d]\n", + netcode_address_to_string( &client->server_address, server_address_string ), client->server_address_index + 1, client->connect_token.num_server_addresses ); + + memcpy( client->context.read_packet_key, client->connect_token.server_to_client_key, NETCODE_KEY_BYTES ); + memcpy( client->context.write_packet_key, client->connect_token.client_to_server_key, NETCODE_KEY_BYTES ); + + netcode_client_reset_before_next_connect( client ); + + netcode_client_set_state( client, NETCODE_CLIENT_STATE_SENDING_CONNECTION_REQUEST ); +} + +void netcode_client_process_packet_internal( struct netcode_client_t * client, struct netcode_address_t * from, uint8_t * packet, uint64_t sequence ) +{ + netcode_assert( client ); + netcode_assert( packet ); + + uint8_t packet_type = ( (uint8_t*) packet ) [0]; + + switch ( packet_type ) + { + case NETCODE_CONNECTION_DENIED_PACKET: + { + if ( ( client->state == NETCODE_CLIENT_STATE_SENDING_CONNECTION_REQUEST || + client->state == NETCODE_CLIENT_STATE_SENDING_CONNECTION_RESPONSE ) + && + netcode_address_equal( from, &client->server_address ) ) + { + client->should_disconnect = 1; + client->should_disconnect_state = NETCODE_CLIENT_STATE_CONNECTION_DENIED; + client->last_packet_receive_time = client->time; + } + } + break; + + case NETCODE_CONNECTION_CHALLENGE_PACKET: + { + if ( client->state == NETCODE_CLIENT_STATE_SENDING_CONNECTION_REQUEST && netcode_address_equal( from, &client->server_address ) ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client received connection challenge packet from server\n" ); + + struct netcode_connection_challenge_packet_t * p = (struct netcode_connection_challenge_packet_t*) packet; + client->challenge_token_sequence = p->challenge_token_sequence; + memcpy( client->challenge_token_data, p->challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES ); + client->last_packet_receive_time = client->time; + + netcode_client_set_state( client, NETCODE_CLIENT_STATE_SENDING_CONNECTION_RESPONSE ); + } + } + break; + + case NETCODE_CONNECTION_KEEP_ALIVE_PACKET: + { + if ( netcode_address_equal( from, &client->server_address ) ) + { + struct netcode_connection_keep_alive_packet_t * p = (struct netcode_connection_keep_alive_packet_t*) packet; + + if ( client->state == NETCODE_CLIENT_STATE_CONNECTED ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client received connection keep alive packet from server\n" ); + + client->last_packet_receive_time = client->time; + } + else if ( client->state == NETCODE_CLIENT_STATE_SENDING_CONNECTION_RESPONSE ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client received connection keep alive packet from server\n" ); + + client->last_packet_receive_time = client->time; + client->client_index = p->client_index; + client->max_clients = p->max_clients; + + netcode_client_set_state( client, NETCODE_CLIENT_STATE_CONNECTED ); + + netcode_printf( NETCODE_LOG_LEVEL_INFO, "client connected to server\n" ); + } + } + } + break; + + case NETCODE_CONNECTION_PAYLOAD_PACKET: + { + if ( client->state == NETCODE_CLIENT_STATE_CONNECTED && netcode_address_equal( from, &client->server_address ) ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client received connection payload packet from server\n" ); + + netcode_packet_queue_push( &client->packet_receive_queue, packet, sequence ); + + client->last_packet_receive_time = client->time; + + return; + } + } + break; + + case NETCODE_CONNECTION_DISCONNECT_PACKET: + { + if ( client->state == NETCODE_CLIENT_STATE_CONNECTED && netcode_address_equal( from, &client->server_address ) ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client received disconnect packet from server\n" ); + + client->should_disconnect = 1; + client->should_disconnect_state = NETCODE_CLIENT_STATE_DISCONNECTED; + client->last_packet_receive_time = client->time; + } + } + break; + + default: + break; + } + + client->config.free_function( client->config.allocator_context, packet ); +} + +void netcode_client_process_packet( struct netcode_client_t * client, struct netcode_address_t * from, uint8_t * packet_data, int packet_bytes ) +{ + (void) client; + (void) from; + (void) packet_data; + (void) packet_bytes; + + uint8_t allowed_packets[NETCODE_CONNECTION_NUM_PACKETS]; + memset( allowed_packets, 0, sizeof( allowed_packets ) ); + allowed_packets[NETCODE_CONNECTION_DENIED_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_CHALLENGE_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_KEEP_ALIVE_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_PAYLOAD_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_DISCONNECT_PACKET] = 1; + + uint64_t current_timestamp = (uint64_t) time( NULL ); + + uint64_t sequence; + + void * packet = netcode_read_packet( packet_data, + packet_bytes, + &sequence, + client->context.read_packet_key, + client->connect_token.protocol_id, + current_timestamp, + NULL, + allowed_packets, + &client->replay_protection, + client->config.allocator_context, + client->config.allocate_function ); + + if ( !packet ) + return; + + netcode_client_process_packet_internal( client, from, (uint8_t*)packet, sequence ); +} + +void netcode_client_receive_packets( struct netcode_client_t * client ) +{ + netcode_assert( client ); + netcode_assert( !client->loopback ); + + uint8_t allowed_packets[NETCODE_CONNECTION_NUM_PACKETS]; + memset( allowed_packets, 0, sizeof( allowed_packets ) ); + allowed_packets[NETCODE_CONNECTION_DENIED_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_CHALLENGE_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_KEEP_ALIVE_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_PAYLOAD_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_DISCONNECT_PACKET] = 1; + + uint64_t current_timestamp = (uint64_t) time( NULL ); + + if ( !client->config.network_simulator ) + { + // process packets received from socket + + while ( 1 ) + { + struct netcode_address_t from; + uint8_t packet_data[NETCODE_MAX_PACKET_BYTES]; + int packet_bytes = 0; + + if ( client->config.override_send_and_receive ) + { + packet_bytes = client->config.receive_packet_override( client->config.callback_context, &from, packet_data, NETCODE_MAX_PACKET_BYTES ); + } + else if ( client->server_address.type == NETCODE_ADDRESS_IPV4 ) + { + packet_bytes = netcode_socket_receive_packet( &client->socket_holder.ipv4, &from, packet_data, NETCODE_MAX_PACKET_BYTES ); + } + else if ( client->server_address.type == NETCODE_ADDRESS_IPV6 ) + { + packet_bytes = netcode_socket_receive_packet( &client->socket_holder.ipv6, &from, packet_data, NETCODE_MAX_PACKET_BYTES ); + } + + if ( packet_bytes == 0 ) + break; + + uint64_t sequence; + void * packet = netcode_read_packet( packet_data, + packet_bytes, + &sequence, + client->context.read_packet_key, + client->connect_token.protocol_id, + current_timestamp, + NULL, + allowed_packets, + &client->replay_protection, + client->config.allocator_context, + client->config.allocate_function ); + + if ( !packet ) + continue; + + netcode_client_process_packet_internal( client, &from, (uint8_t*)packet, sequence ); + } + } + else + { + // process packets received from network simulator + + int num_packets_received = netcode_network_simulator_receive_packets( client->config.network_simulator, + &client->address, + NETCODE_CLIENT_MAX_RECEIVE_PACKETS, + client->receive_packet_data, + client->receive_packet_bytes, + client->receive_from ); + + int i; + for ( i = 0; i < num_packets_received; ++i ) + { + uint64_t sequence; + + void * packet = netcode_read_packet( client->receive_packet_data[i], + client->receive_packet_bytes[i], + &sequence, + client->context.read_packet_key, + client->connect_token.protocol_id, + current_timestamp, + NULL, + allowed_packets, + &client->replay_protection, + client->config.allocator_context, + client->config.allocate_function ); + + client->config.free_function( client->config.allocator_context, client->receive_packet_data[i] ); + + if ( !packet ) + continue; + + netcode_client_process_packet_internal( client, &client->receive_from[i], (uint8_t*)packet, sequence ); + } + } +} + +void netcode_client_send_packet_to_server_internal( struct netcode_client_t * client, void * packet ) +{ + netcode_assert( client ); + netcode_assert( !client->loopback ); + + uint8_t packet_data[NETCODE_MAX_PACKET_BYTES]; + + int packet_bytes = netcode_write_packet( packet, + packet_data, + NETCODE_MAX_PACKET_BYTES, + client->sequence++, + client->context.write_packet_key, + client->connect_token.protocol_id ); + + netcode_assert( packet_bytes <= NETCODE_MAX_PACKET_BYTES ); + + if ( client->config.network_simulator ) + { + netcode_network_simulator_send_packet( client->config.network_simulator, &client->address, &client->server_address, packet_data, packet_bytes ); + } + else + { + if ( client->config.override_send_and_receive ) + { + client->config.send_packet_override( client->config.callback_context, &client->server_address, packet_data, packet_bytes ); + } + else if ( client->server_address.type == NETCODE_ADDRESS_IPV4 ) + { + netcode_socket_send_packet( &client->socket_holder.ipv4, &client->server_address, packet_data, packet_bytes ); + } + else if ( client->server_address.type == NETCODE_ADDRESS_IPV6 ) + { + netcode_socket_send_packet( &client->socket_holder.ipv6, &client->server_address, packet_data, packet_bytes ); + } + } + + client->last_packet_send_time = client->time; +} + +void netcode_client_send_packets( struct netcode_client_t * client ) +{ + netcode_assert( client ); + netcode_assert( !client->loopback ); + + switch ( client->state ) + { + case NETCODE_CLIENT_STATE_SENDING_CONNECTION_REQUEST: + { + if ( client->last_packet_send_time + ( 1.0 / NETCODE_PACKET_SEND_RATE ) >= client->time ) + return; + + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client sent connection request packet to server\n" ); + + struct netcode_connection_request_packet_t packet; + packet.packet_type = NETCODE_CONNECTION_REQUEST_PACKET; + memcpy( packet.version_info, NETCODE_VERSION_INFO, NETCODE_VERSION_INFO_BYTES ); + packet.protocol_id = client->connect_token.protocol_id; + packet.connect_token_expire_timestamp = client->connect_token.expire_timestamp; + memcpy( packet.connect_token_nonce, client->connect_token.nonce, NETCODE_CONNECT_TOKEN_NONCE_BYTES ); + memcpy( packet.connect_token_data, client->connect_token.private_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + netcode_client_send_packet_to_server_internal( client, &packet ); + } + break; + + case NETCODE_CLIENT_STATE_SENDING_CONNECTION_RESPONSE: + { + if ( client->last_packet_send_time + ( 1.0 / NETCODE_PACKET_SEND_RATE ) >= client->time ) + return; + + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client sent connection response packet to server\n" ); + + struct netcode_connection_response_packet_t packet; + packet.packet_type = NETCODE_CONNECTION_RESPONSE_PACKET; + packet.challenge_token_sequence = client->challenge_token_sequence; + memcpy( packet.challenge_token_data, client->challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES ); + + netcode_client_send_packet_to_server_internal( client, &packet ); + } + break; + + case NETCODE_CLIENT_STATE_CONNECTED: + { + if ( client->last_packet_send_time + ( 1.0 / NETCODE_PACKET_SEND_RATE ) >= client->time ) + return; + + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client sent connection keep-alive packet to server\n" ); + + struct netcode_connection_keep_alive_packet_t packet; + packet.packet_type = NETCODE_CONNECTION_KEEP_ALIVE_PACKET; + packet.client_index = 0; + packet.max_clients = 0; + + netcode_client_send_packet_to_server_internal( client, &packet ); + } + break; + + default: + break; + } +} + +int netcode_client_connect_to_next_server( struct netcode_client_t * client ) +{ + netcode_assert( client ); + + if ( client->server_address_index + 1 >= client->connect_token.num_server_addresses ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client has no more servers to connect to\n" ); + return 0; + } + + client->server_address_index++; + client->server_address = client->connect_token.server_addresses[client->server_address_index]; + + netcode_client_reset_before_next_connect( client ); + + char server_address_string[NETCODE_MAX_ADDRESS_STRING_LENGTH]; + + netcode_printf( NETCODE_LOG_LEVEL_INFO, "client connecting to next server %s [%d/%d]\n", + netcode_address_to_string( &client->server_address, server_address_string ), + client->server_address_index + 1, + client->connect_token.num_server_addresses ); + + netcode_client_set_state( client, NETCODE_CLIENT_STATE_SENDING_CONNECTION_REQUEST ); + + return 1; +} + +void netcode_client_update( struct netcode_client_t * client, double time ) +{ + netcode_assert( client ); + + client->time = time; + + if ( client->loopback ) + return; + + netcode_client_receive_packets( client ); + + netcode_client_send_packets( client ); + + if ( client->state > NETCODE_CLIENT_STATE_DISCONNECTED && client->state < NETCODE_CLIENT_STATE_CONNECTED ) + { + uint64_t connect_token_expire_seconds = ( client->connect_token.expire_timestamp - client->connect_token.create_timestamp ); + if ( client->time - client->connect_start_time >= connect_token_expire_seconds ) + { + netcode_printf( NETCODE_LOG_LEVEL_INFO, "client connect failed. connect token expired\n" ); + netcode_client_disconnect_internal( client, NETCODE_CLIENT_STATE_CONNECT_TOKEN_EXPIRED, 0 ); + return; + } + } + + if ( client->should_disconnect ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client should disconnect -> %s\n", netcode_client_state_name( client->should_disconnect_state ) ); + if ( netcode_client_connect_to_next_server( client ) ) + return; + netcode_client_disconnect_internal( client, client->should_disconnect_state, 0 ); + return; + } + + switch ( client->state ) + { + case NETCODE_CLIENT_STATE_SENDING_CONNECTION_REQUEST: + { + if ( client->connect_token.timeout_seconds > 0 && client->last_packet_receive_time + client->connect_token.timeout_seconds < time ) + { + netcode_printf( NETCODE_LOG_LEVEL_INFO, "client connect failed. connection request timed out\n" ); + if ( netcode_client_connect_to_next_server( client ) ) + return; + netcode_client_disconnect_internal( client, NETCODE_CLIENT_STATE_CONNECTION_REQUEST_TIMED_OUT, 0 ); + return; + } + } + break; + + case NETCODE_CLIENT_STATE_SENDING_CONNECTION_RESPONSE: + { + if ( client->connect_token.timeout_seconds > 0 && client->last_packet_receive_time + client->connect_token.timeout_seconds < time ) + { + netcode_printf( NETCODE_LOG_LEVEL_INFO, "client connect failed. connection response timed out\n" ); + if ( netcode_client_connect_to_next_server( client ) ) + return; + netcode_client_disconnect_internal( client, NETCODE_CLIENT_STATE_CONNECTION_RESPONSE_TIMED_OUT, 0 ); + return; + } + } + break; + + case NETCODE_CLIENT_STATE_CONNECTED: + { + if ( client->connect_token.timeout_seconds > 0 && client->last_packet_receive_time + client->connect_token.timeout_seconds < time ) + { + netcode_printf( NETCODE_LOG_LEVEL_INFO, "client connection timed out\n" ); + netcode_client_disconnect_internal( client, NETCODE_CLIENT_STATE_CONNECTION_TIMED_OUT, 0 ); + return; + } + } + break; + + default: + break; + } +} + +uint64_t netcode_client_next_packet_sequence( struct netcode_client_t * client ) +{ + netcode_assert( client ); + return client->sequence; +} + +void netcode_client_send_packet( struct netcode_client_t * client, NETCODE_CONST uint8_t * packet_data, int packet_bytes ) +{ + netcode_assert( client ); + netcode_assert( packet_data ); + netcode_assert( packet_bytes >= 0 ); + netcode_assert( packet_bytes <= NETCODE_MAX_PACKET_SIZE ); + + if ( client->state != NETCODE_CLIENT_STATE_CONNECTED ) + return; + + if ( !client->loopback ) + { + uint8_t buffer[NETCODE_MAX_PAYLOAD_BYTES*2]; + + struct netcode_connection_payload_packet_t * packet = (struct netcode_connection_payload_packet_t*) buffer; + + packet->packet_type = NETCODE_CONNECTION_PAYLOAD_PACKET; + packet->payload_bytes = packet_bytes; + memcpy( packet->payload_data, packet_data, packet_bytes ); + + netcode_client_send_packet_to_server_internal( client, packet ); + } + else + { + client->config.send_loopback_packet_callback( client->config.callback_context, + client->client_index, + packet_data, + packet_bytes, + client->sequence++ ); + } +} + +uint8_t * netcode_client_receive_packet( struct netcode_client_t * client, int * packet_bytes, uint64_t * packet_sequence ) +{ + netcode_assert( client ); + netcode_assert( packet_bytes ); + + struct netcode_connection_payload_packet_t * packet = (struct netcode_connection_payload_packet_t*) + netcode_packet_queue_pop( &client->packet_receive_queue, packet_sequence ); + + if ( packet ) + { + netcode_assert( packet->packet_type == NETCODE_CONNECTION_PAYLOAD_PACKET ); + *packet_bytes = packet->payload_bytes; + netcode_assert( *packet_bytes >= 0 ); + netcode_assert( *packet_bytes <= NETCODE_MAX_PAYLOAD_BYTES ); + return (uint8_t*) &packet->payload_data; + } + else + { + return NULL; + } +} + +void netcode_client_free_packet( struct netcode_client_t * client, void * packet ) +{ + netcode_assert( client ); + netcode_assert( packet ); + uint8_t * packet_data = (uint8_t*) packet; + int offset = offsetof( struct netcode_connection_payload_packet_t, payload_data ); + client->config.free_function( client->config.allocator_context, packet_data - offset ); +} + +void netcode_client_disconnect( struct netcode_client_t * client ) +{ + netcode_assert( client ); + netcode_assert( !client->loopback ); + netcode_client_disconnect_internal( client, NETCODE_CLIENT_STATE_DISCONNECTED, 1 ); +} + +void netcode_client_disconnect_internal( struct netcode_client_t * client, int destination_state, int send_disconnect_packets ) +{ + netcode_assert( !client->loopback ); + netcode_assert( destination_state <= NETCODE_CLIENT_STATE_DISCONNECTED ); + + if ( client->state <= NETCODE_CLIENT_STATE_DISCONNECTED || client->state == destination_state ) + return; + + netcode_printf( NETCODE_LOG_LEVEL_INFO, "client disconnected\n" ); + + if ( !client->loopback && send_disconnect_packets && client->state > NETCODE_CLIENT_STATE_DISCONNECTED ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client sent disconnect packets to server\n" ); + + int i; + for ( i = 0; i < NETCODE_NUM_DISCONNECT_PACKETS; ++i ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client sent disconnect packet %d\n", i ); + + struct netcode_connection_disconnect_packet_t packet; + packet.packet_type = NETCODE_CONNECTION_DISCONNECT_PACKET; + + netcode_client_send_packet_to_server_internal( client, &packet ); + } + } + + netcode_client_reset_connection_data( client, destination_state ); +} + +int netcode_client_state( struct netcode_client_t * client ) +{ + netcode_assert( client ); + return client->state; +} + +int netcode_client_index( struct netcode_client_t * client ) +{ + netcode_assert( client ); + return client->client_index; +} + +int netcode_client_max_clients( struct netcode_client_t * client ) +{ + netcode_assert( client ); + return client->max_clients; +} + +void netcode_client_connect_loopback( struct netcode_client_t * client, int client_index, int max_clients ) +{ + netcode_assert( client ); + netcode_assert( client->state <= NETCODE_CLIENT_STATE_DISCONNECTED ); + netcode_printf( NETCODE_LOG_LEVEL_INFO, "client connected to server via loopback as client %d\n", client_index ); + client->state = NETCODE_CLIENT_STATE_CONNECTED; + client->client_index = client_index; + client->max_clients = max_clients; + client->loopback = 1; +} + +void netcode_client_disconnect_loopback( struct netcode_client_t * client ) +{ + netcode_assert( client ); + netcode_assert( client->loopback ); + netcode_client_reset_connection_data( client, NETCODE_CLIENT_STATE_DISCONNECTED ); +} + +int netcode_client_loopback( struct netcode_client_t * client ) +{ + netcode_assert( client ); + return client->loopback; +} + +void netcode_client_process_loopback_packet( struct netcode_client_t * client, NETCODE_CONST uint8_t * packet_data, int packet_bytes, uint64_t packet_sequence ) +{ + netcode_assert( client ); + netcode_assert( client->loopback ); + struct netcode_connection_payload_packet_t * packet = netcode_create_payload_packet( packet_bytes, client->config.allocator_context, client->config.allocate_function ); + if ( !packet ) + return; + memcpy( packet->payload_data, packet_data, packet_bytes ); + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "client processing loopback packet from server\n" ); + netcode_packet_queue_push( &client->packet_receive_queue, packet, packet_sequence ); +} + +uint16_t netcode_client_get_port( struct netcode_client_t * client ) +{ + netcode_assert( client ); + return client->address.type == NETCODE_ADDRESS_IPV4 ? client->socket_holder.ipv4.address.port : client->socket_holder.ipv6.address.port; +} + +struct netcode_address_t * netcode_client_server_address( struct netcode_client_t * client ) +{ + netcode_assert( client ); + return &client->server_address; +} + +// ---------------------------------------------------------------- + +#define NETCODE_MAX_ENCRYPTION_MAPPINGS ( NETCODE_MAX_CLIENTS * 4 ) + +struct netcode_encryption_manager_t +{ + int num_encryption_mappings; + int timeout[NETCODE_MAX_ENCRYPTION_MAPPINGS]; + double expire_time[NETCODE_MAX_ENCRYPTION_MAPPINGS]; + double last_access_time[NETCODE_MAX_ENCRYPTION_MAPPINGS]; + struct netcode_address_t address[NETCODE_MAX_ENCRYPTION_MAPPINGS]; + int client_index[NETCODE_MAX_ENCRYPTION_MAPPINGS]; + uint8_t send_key[NETCODE_KEY_BYTES*NETCODE_MAX_ENCRYPTION_MAPPINGS]; + uint8_t receive_key[NETCODE_KEY_BYTES*NETCODE_MAX_ENCRYPTION_MAPPINGS]; +}; + +void netcode_encryption_manager_reset( struct netcode_encryption_manager_t * encryption_manager ) +{ + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "reset encryption manager\n" ); + + netcode_assert( encryption_manager ); + + encryption_manager->num_encryption_mappings = 0; + + int i; + for ( i = 0; i < NETCODE_MAX_ENCRYPTION_MAPPINGS; ++i ) + { + encryption_manager->client_index[i] = -1; + encryption_manager->expire_time[i] = -1.0; + encryption_manager->last_access_time[i] = -1000.0; + memset( &encryption_manager->address[i], 0, sizeof( struct netcode_address_t ) ); + } + + memset( encryption_manager->timeout, 0, sizeof( encryption_manager->timeout ) ); + memset( encryption_manager->send_key, 0, sizeof( encryption_manager->send_key ) ); + memset( encryption_manager->receive_key, 0, sizeof( encryption_manager->receive_key ) ); +} + +int netcode_encryption_manager_entry_expired( struct netcode_encryption_manager_t * encryption_manager, int index, double time ) +{ + return ( encryption_manager->timeout[index] > 0 && ( encryption_manager->last_access_time[index] + encryption_manager->timeout[index] ) < time ) || + ( encryption_manager->expire_time[index] >= 0.0 && encryption_manager->expire_time[index] < time ); +} + +int netcode_encryption_manager_add_encryption_mapping( struct netcode_encryption_manager_t * encryption_manager, + struct netcode_address_t * address, + uint8_t * send_key, + uint8_t * receive_key, + double time, + double expire_time, + int timeout ) +{ + int i; + for ( i = 0; i < encryption_manager->num_encryption_mappings; ++i ) + { + if ( netcode_address_equal( &encryption_manager->address[i], address ) && !netcode_encryption_manager_entry_expired( encryption_manager, i, time ) ) + { + encryption_manager->timeout[i] = timeout; + encryption_manager->expire_time[i] = expire_time; + encryption_manager->last_access_time[i] = time; + memcpy( encryption_manager->send_key + i * NETCODE_KEY_BYTES, send_key, NETCODE_KEY_BYTES ); + memcpy( encryption_manager->receive_key + i * NETCODE_KEY_BYTES, receive_key, NETCODE_KEY_BYTES ); + return 1; + } + } + + for ( i = 0; i < NETCODE_MAX_ENCRYPTION_MAPPINGS; ++i ) + { + if ( encryption_manager->address[i].type == NETCODE_ADDRESS_NONE || + ( netcode_encryption_manager_entry_expired( encryption_manager, i, time ) && encryption_manager->client_index[i] == -1 ) ) + { + encryption_manager->timeout[i] = timeout; + encryption_manager->address[i] = *address; + encryption_manager->expire_time[i] = expire_time; + encryption_manager->last_access_time[i] = time; + memcpy( encryption_manager->send_key + i * NETCODE_KEY_BYTES, send_key, NETCODE_KEY_BYTES ); + memcpy( encryption_manager->receive_key + i * NETCODE_KEY_BYTES, receive_key, NETCODE_KEY_BYTES ); + if ( i + 1 > encryption_manager->num_encryption_mappings ) + encryption_manager->num_encryption_mappings = i + 1; + return 1; + } + } + + return 0; +} + +int netcode_encryption_manager_remove_encryption_mapping( struct netcode_encryption_manager_t * encryption_manager, struct netcode_address_t * address, double time ) +{ + netcode_assert( encryption_manager ); + netcode_assert( address ); + + int i; + for ( i = 0; i < encryption_manager->num_encryption_mappings; ++i ) + { + if ( netcode_address_equal( &encryption_manager->address[i], address ) ) + { + encryption_manager->expire_time[i] = -1.0; + encryption_manager->last_access_time[i] = -1000.0; + memset( &encryption_manager->address[i], 0, sizeof( struct netcode_address_t ) ); + memset( encryption_manager->send_key + i * NETCODE_KEY_BYTES, 0, NETCODE_KEY_BYTES ); + memset( encryption_manager->receive_key + i * NETCODE_KEY_BYTES, 0, NETCODE_KEY_BYTES ); + + if ( i + 1 == encryption_manager->num_encryption_mappings ) + { + int index = i - 1; + while ( index >= 0 ) + { + if ( !netcode_encryption_manager_entry_expired( encryption_manager, index, time ) || encryption_manager->client_index[index] != -1 ) + { + break; + } + encryption_manager->address[index].type = NETCODE_ADDRESS_NONE; + index--; + } + encryption_manager->num_encryption_mappings = index + 1; + } + + return 1; + } + } + + return 0; +} + +int netcode_encryption_manager_find_encryption_mapping( struct netcode_encryption_manager_t * encryption_manager, struct netcode_address_t * address, double time ) +{ + int i; + for ( i = 0; i < encryption_manager->num_encryption_mappings; ++i ) + { + if ( netcode_address_equal( &encryption_manager->address[i], address ) && !netcode_encryption_manager_entry_expired( encryption_manager, i, time ) ) + { + encryption_manager->last_access_time[i] = time; + return i; + } + } + return -1; +} + +int netcode_encryption_manager_touch( struct netcode_encryption_manager_t * encryption_manager, int index, struct netcode_address_t * address, double time ) +{ + netcode_assert( index >= 0 ); + netcode_assert( index < encryption_manager->num_encryption_mappings ); + if ( !netcode_address_equal( &encryption_manager->address[index], address ) ) + return 0; + encryption_manager->last_access_time[index] = time; + return 1; +} + +void netcode_encryption_manager_set_expire_time( struct netcode_encryption_manager_t * encryption_manager, int index, double expire_time ) +{ + netcode_assert( index >= 0 ); + netcode_assert( index < encryption_manager->num_encryption_mappings ); + encryption_manager->expire_time[index] = expire_time; +} + + +uint8_t * netcode_encryption_manager_get_send_key( struct netcode_encryption_manager_t * encryption_manager, int index ) +{ + netcode_assert( encryption_manager ); + if ( index == -1 ) + return NULL; + netcode_assert( index >= 0 ); + netcode_assert( index < encryption_manager->num_encryption_mappings ); + return encryption_manager->send_key + index * NETCODE_KEY_BYTES; +} + +uint8_t * netcode_encryption_manager_get_receive_key( struct netcode_encryption_manager_t * encryption_manager, int index ) +{ + netcode_assert( encryption_manager ); + if ( index == -1 ) + return NULL; + netcode_assert( index >= 0 ); + netcode_assert( index < encryption_manager->num_encryption_mappings ); + return encryption_manager->receive_key + index * NETCODE_KEY_BYTES; +} + +int netcode_encryption_manager_get_timeout( struct netcode_encryption_manager_t * encryption_manager, int index ) +{ + netcode_assert( encryption_manager ); + if ( index == -1 ) + return 0; + netcode_assert( index >= 0 ); + netcode_assert( index < encryption_manager->num_encryption_mappings ); + return encryption_manager->timeout[index]; +} + +// ---------------------------------------------------------------- + +#define NETCODE_MAX_CONNECT_TOKEN_ENTRIES ( NETCODE_MAX_CLIENTS * 8 ) + +struct netcode_connect_token_entry_t +{ + double time; + uint8_t mac[NETCODE_MAC_BYTES]; + struct netcode_address_t address; +}; + +void netcode_connect_token_entries_reset( struct netcode_connect_token_entry_t * connect_token_entries ) +{ + int i; + for ( i = 0; i < NETCODE_MAX_CONNECT_TOKEN_ENTRIES; ++i ) + { + connect_token_entries[i].time = -1000.0; + memset( connect_token_entries[i].mac, 0, NETCODE_MAC_BYTES ); + memset( &connect_token_entries[i].address, 0, sizeof( struct netcode_address_t ) ); + } +} + +int netcode_connect_token_entries_find_or_add( struct netcode_connect_token_entry_t * connect_token_entries, + struct netcode_address_t * address, + uint8_t * mac, + double time ) +{ + netcode_assert( connect_token_entries ); + netcode_assert( address ); + netcode_assert( mac ); + + // find the matching entry for the token mac and the oldest token entry. constant time worst case. This is intentional! + + int matching_token_index = -1; + int oldest_token_index = -1; + double oldest_token_time = 0.0; + + int i; + for ( i = 0; i < NETCODE_MAX_CONNECT_TOKEN_ENTRIES; ++i ) + { + if ( memcmp( mac, connect_token_entries[i].mac, NETCODE_MAC_BYTES ) == 0 ) + matching_token_index = i; + + if ( oldest_token_index == -1 || connect_token_entries[i].time < oldest_token_time ) + { + oldest_token_time = connect_token_entries[i].time; + oldest_token_index = i; + } + } + + // if no entry is found with the mac, this is a new connect token. replace the oldest token entry. + + netcode_assert( oldest_token_index != -1 ); + + if ( matching_token_index == -1 ) + { + connect_token_entries[oldest_token_index].time = time; + connect_token_entries[oldest_token_index].address = *address; + memcpy( connect_token_entries[oldest_token_index].mac, mac, NETCODE_MAC_BYTES ); + return 1; + } + + // allow connect tokens we have already seen from the same address + + netcode_assert( matching_token_index >= 0 ); + netcode_assert( matching_token_index < NETCODE_MAX_CONNECT_TOKEN_ENTRIES ); + if ( netcode_address_equal( &connect_token_entries[matching_token_index].address, address ) ) + return 1; + + return 0; +} + +// ---------------------------------------------------------------- + +#define NETCODE_SERVER_FLAG_IGNORE_CONNECTION_REQUEST_PACKETS 1 +#define NETCODE_SERVER_FLAG_IGNORE_CONNECTION_RESPONSE_PACKETS (1<<1) + +void netcode_default_server_config( struct netcode_server_config_t * config ) +{ + netcode_assert( config ); + config->allocator_context = NULL; + config->allocate_function = netcode_default_allocate_function; + config->free_function = netcode_default_free_function; + config->network_simulator = NULL; + config->callback_context = NULL; + config->connect_disconnect_callback = NULL; + config->send_loopback_packet_callback = NULL; + config->override_send_and_receive = 0; + config->send_packet_override = NULL; + config->receive_packet_override = NULL; +}; + +struct netcode_server_t +{ + struct netcode_server_config_t config; + struct netcode_socket_holder_t socket_holder; + struct netcode_address_t address; + uint32_t flags; + double time; + int running; + int max_clients; + int num_connected_clients; + uint64_t global_sequence; + uint64_t challenge_sequence; + uint8_t challenge_key[NETCODE_KEY_BYTES]; + int client_connected[NETCODE_MAX_CLIENTS]; + int client_timeout[NETCODE_MAX_CLIENTS]; + int client_loopback[NETCODE_MAX_CLIENTS]; + int client_confirmed[NETCODE_MAX_CLIENTS]; + int client_encryption_index[NETCODE_MAX_CLIENTS]; + uint64_t client_id[NETCODE_MAX_CLIENTS]; + uint64_t client_sequence[NETCODE_MAX_CLIENTS]; + double client_last_packet_send_time[NETCODE_MAX_CLIENTS]; + double client_last_packet_receive_time[NETCODE_MAX_CLIENTS]; + uint8_t client_user_data[NETCODE_MAX_CLIENTS][NETCODE_USER_DATA_BYTES]; + struct netcode_replay_protection_t client_replay_protection[NETCODE_MAX_CLIENTS]; + struct netcode_packet_queue_t client_packet_queue[NETCODE_MAX_CLIENTS]; + struct netcode_address_t client_address[NETCODE_MAX_CLIENTS]; + struct netcode_connect_token_entry_t connect_token_entries[NETCODE_MAX_CONNECT_TOKEN_ENTRIES]; + struct netcode_encryption_manager_t encryption_manager; + uint8_t * receive_packet_data[NETCODE_SERVER_MAX_RECEIVE_PACKETS]; + int receive_packet_bytes[NETCODE_SERVER_MAX_RECEIVE_PACKETS]; + struct netcode_address_t receive_from[NETCODE_SERVER_MAX_RECEIVE_PACKETS]; +}; + +int netcode_server_socket_create( struct netcode_socket_t * socket, + struct netcode_address_t * address, + int send_buffer_size, + int receive_buffer_size, + NETCODE_CONST struct netcode_server_config_t * config ) +{ + netcode_assert( socket ); + netcode_assert( address ); + netcode_assert( config ); + + if ( !config->network_simulator ) + { + if ( !config->override_send_and_receive ) + { + if ( netcode_socket_create( socket, address, send_buffer_size, receive_buffer_size ) != NETCODE_SOCKET_ERROR_NONE ) + { + return 0; + } + } + } + + return 1; +} + +struct netcode_server_t * netcode_server_create_overload( NETCODE_CONST char * server_address1_string, NETCODE_CONST char * server_address2_string, NETCODE_CONST struct netcode_server_config_t * config, double time ) +{ + netcode_assert( config ); + netcode_assert( netcode.initialized ); + + struct netcode_address_t server_address1; + struct netcode_address_t server_address2; + + memset( &server_address1, 0, sizeof( server_address1 ) ); + memset( &server_address2, 0, sizeof( server_address2 ) ); + + if ( netcode_parse_address( server_address1_string, &server_address1 ) != NETCODE_OK ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to parse server public address\n" ); + return NULL; + } + + if ( server_address2_string != NULL && netcode_parse_address( server_address2_string, &server_address2 ) != NETCODE_OK ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: failed to parse server public address2\n" ); + return NULL; + } + + struct netcode_address_t bind_address_ipv4; + struct netcode_address_t bind_address_ipv6; + + memset( &bind_address_ipv4, 0, sizeof( bind_address_ipv4 ) ); + memset( &bind_address_ipv6, 0, sizeof( bind_address_ipv6 ) ); + + struct netcode_socket_t socket_ipv4; + struct netcode_socket_t socket_ipv6; + + memset( &socket_ipv4, 0, sizeof( socket_ipv4 ) ); + memset( &socket_ipv6, 0, sizeof( socket_ipv6 ) ); + + if ( server_address1.type == NETCODE_ADDRESS_IPV4 || server_address2.type == NETCODE_ADDRESS_IPV4 ) + { + bind_address_ipv4.type = NETCODE_ADDRESS_IPV4; + bind_address_ipv4.port = server_address1.type == NETCODE_ADDRESS_IPV4 ? server_address1.port : server_address2.port; + + if ( !netcode_server_socket_create( &socket_ipv4, &bind_address_ipv4, NETCODE_SERVER_SOCKET_SNDBUF_SIZE, NETCODE_SERVER_SOCKET_RCVBUF_SIZE, config ) ) + { + return NULL; + } + } + + if ( server_address1.type == NETCODE_ADDRESS_IPV6 || server_address2.type == NETCODE_ADDRESS_IPV6 ) + { + bind_address_ipv6.type = NETCODE_ADDRESS_IPV6; + bind_address_ipv6.port = server_address1.type == NETCODE_ADDRESS_IPV6 ? server_address1.port : server_address2.port; + + if ( !netcode_server_socket_create( &socket_ipv6, &bind_address_ipv6, NETCODE_SERVER_SOCKET_SNDBUF_SIZE, NETCODE_SERVER_SOCKET_RCVBUF_SIZE, config ) ) + { + return NULL; + } + } + + struct netcode_server_t * server = (struct netcode_server_t*) config->allocate_function( config->allocator_context, sizeof( struct netcode_server_t ) ); + if ( !server ) + { + netcode_socket_destroy( &socket_ipv4 ); + netcode_socket_destroy( &socket_ipv6 ); + return NULL; + } + + if ( !config->network_simulator ) + { + netcode_printf( NETCODE_LOG_LEVEL_INFO, "server listening on %s\n", server_address1_string ); + } + else + { + netcode_printf( NETCODE_LOG_LEVEL_INFO, "server listening on %s (network simulator)\n", server_address1_string ); + } + + server->config = *config; + server->socket_holder.ipv4 = socket_ipv4; + server->socket_holder.ipv6 = socket_ipv6; + server->address = server_address1; + server->flags = 0; + server->time = time; + server->running = 0; + server->max_clients = 0; + server->num_connected_clients = 0; + server->global_sequence = 1ULL << 63; + + memset( server->client_connected, 0, sizeof( server->client_connected ) ); + memset( server->client_loopback, 0, sizeof( server->client_loopback ) ); + memset( server->client_confirmed, 0, sizeof( server->client_confirmed ) ); + memset( server->client_id, 0, sizeof( server->client_id ) ); + memset( server->client_sequence, 0, sizeof( server->client_sequence ) ); + memset( server->client_last_packet_send_time, 0, sizeof( server->client_last_packet_send_time ) ); + memset( server->client_last_packet_receive_time, 0, sizeof( server->client_last_packet_receive_time ) ); + memset( server->client_address, 0, sizeof( server->client_address ) ); + memset( server->client_user_data, 0, sizeof( server->client_user_data ) ); + + int i; + for ( i = 0; i < NETCODE_MAX_CLIENTS; ++i ) + server->client_encryption_index[i] = -1; + + netcode_connect_token_entries_reset( server->connect_token_entries ); + + netcode_encryption_manager_reset( &server->encryption_manager ); + + for ( i = 0; i < NETCODE_MAX_CLIENTS; ++i ) + netcode_replay_protection_reset( &server->client_replay_protection[i] ); + + memset( &server->client_packet_queue, 0, sizeof( server->client_packet_queue ) ); + + return server; +} + +struct netcode_server_t * netcode_server_create( NETCODE_CONST char * server_address_string, NETCODE_CONST struct netcode_server_config_t * config, double time ) +{ + return netcode_server_create_overload( server_address_string, NULL, config, time ); +} + +void netcode_server_stop( struct netcode_server_t * server ); + +void netcode_server_destroy( struct netcode_server_t * server ) +{ + netcode_assert( server ); + + netcode_server_stop( server ); + + netcode_socket_destroy( &server->socket_holder.ipv4 ); + netcode_socket_destroy( &server->socket_holder.ipv6 ); + + server->config.free_function( server->config.allocator_context, server ); +} + +void netcode_server_start( struct netcode_server_t * server, int max_clients ) +{ + netcode_assert( server ); + netcode_assert( max_clients > 0 ); + netcode_assert( max_clients <= NETCODE_MAX_CLIENTS ); + + if ( server->running ) + netcode_server_stop( server ); + + netcode_printf( NETCODE_LOG_LEVEL_INFO, "server started with %d client slots\n", max_clients ); + + server->running = 1; + server->max_clients = max_clients; + server->num_connected_clients = 0; + server->challenge_sequence = 0; + netcode_generate_key( server->challenge_key ); + + int i; + for ( i = 0; i < server->max_clients; ++i ) + { + netcode_packet_queue_init( &server->client_packet_queue[i], server->config.allocator_context, server->config.allocate_function, server->config.free_function ); + } +} + +void netcode_server_send_global_packet( struct netcode_server_t * server, void * packet, struct netcode_address_t * to, uint8_t * packet_key ) +{ + netcode_assert( server ); + netcode_assert( packet ); + netcode_assert( to ); + netcode_assert( packet_key ); + + uint8_t packet_data[NETCODE_MAX_PACKET_BYTES]; + + int packet_bytes = netcode_write_packet( packet, packet_data, NETCODE_MAX_PACKET_BYTES, server->global_sequence, packet_key, server->config.protocol_id ); + + netcode_assert( packet_bytes <= NETCODE_MAX_PACKET_BYTES ); + + if ( server->config.network_simulator ) + { + netcode_network_simulator_send_packet( server->config.network_simulator, &server->address, to, packet_data, packet_bytes ); + } + else + { + if ( server->config.override_send_and_receive ) + { + server->config.send_packet_override( server->config.callback_context, to, packet_data, packet_bytes ); + } + else if ( to->type == NETCODE_ADDRESS_IPV4 ) + { + netcode_socket_send_packet( &server->socket_holder.ipv4, to, packet_data, packet_bytes ); + } + else if ( to->type == NETCODE_ADDRESS_IPV6 ) + { + netcode_socket_send_packet( &server->socket_holder.ipv6, to, packet_data, packet_bytes ); + } + } + + server->global_sequence++; +} + +void netcode_server_send_client_packet( struct netcode_server_t * server, void * packet, int client_index ) +{ + netcode_assert( server ); + netcode_assert( packet ); + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + netcode_assert( server->client_connected[client_index] ); + netcode_assert( !server->client_loopback[client_index] ); + + uint8_t packet_data[NETCODE_MAX_PACKET_BYTES]; + + if ( !netcode_encryption_manager_touch( &server->encryption_manager, + server->client_encryption_index[client_index], + &server->client_address[client_index], + server->time ) ) + { + netcode_printf( NETCODE_LOG_LEVEL_ERROR, "error: encryption mapping is out of date for client %d\n", client_index ); + return; + } + + uint8_t * packet_key = netcode_encryption_manager_get_send_key( &server->encryption_manager, server->client_encryption_index[client_index] ); + + int packet_bytes = netcode_write_packet( packet, packet_data, NETCODE_MAX_PACKET_BYTES, server->client_sequence[client_index], packet_key, server->config.protocol_id ); + + netcode_assert( packet_bytes <= NETCODE_MAX_PACKET_BYTES ); + + if ( server->config.network_simulator ) + { + netcode_network_simulator_send_packet( server->config.network_simulator, &server->address, &server->client_address[client_index], packet_data, packet_bytes ); + } + else + { + if ( server->config.override_send_and_receive ) + { + server->config.send_packet_override( server->config.callback_context, &server->client_address[client_index], packet_data, packet_bytes ); + } + else + { + if ( server->client_address[client_index].type == NETCODE_ADDRESS_IPV4 ) + { + netcode_socket_send_packet( &server->socket_holder.ipv4, &server->client_address[client_index], packet_data, packet_bytes ); + } + else if ( server->client_address[client_index].type == NETCODE_ADDRESS_IPV6 ) + { + netcode_socket_send_packet( &server->socket_holder.ipv6, &server->client_address[client_index], packet_data, packet_bytes ); + } + } + } + + server->client_sequence[client_index]++; + + server->client_last_packet_send_time[client_index] = server->time; +} + +void netcode_server_disconnect_client_internal( struct netcode_server_t * server, int client_index, int send_disconnect_packets ) +{ + netcode_assert( server ); + netcode_assert( server->running ); + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + netcode_assert( server->client_connected[client_index] ); + netcode_assert( !server->client_loopback[client_index] ); + netcode_assert( server->encryption_manager.client_index[server->client_encryption_index[client_index]] == client_index ); + + netcode_printf( NETCODE_LOG_LEVEL_INFO, "server disconnected client %d\n", client_index ); + + if ( server->config.connect_disconnect_callback ) + { + server->config.connect_disconnect_callback( server->config.callback_context, client_index, 0 ); + } + + if ( send_disconnect_packets ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server sent disconnect packets to client %d\n", client_index ); + + int i; + for ( i = 0; i < NETCODE_NUM_DISCONNECT_PACKETS; ++i ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server sent disconnect packet %d\n", i ); + + struct netcode_connection_disconnect_packet_t packet; + packet.packet_type = NETCODE_CONNECTION_DISCONNECT_PACKET; + + netcode_server_send_client_packet( server, &packet, client_index ); + } + } + + while ( 1 ) + { + void * packet = netcode_packet_queue_pop( &server->client_packet_queue[client_index], NULL ); + if ( !packet ) + break; + server->config.free_function( server->config.allocator_context, packet ); + } + + netcode_packet_queue_clear( &server->client_packet_queue[client_index] ); + + netcode_replay_protection_reset( &server->client_replay_protection[client_index] ); + + server->encryption_manager.client_index[server->client_encryption_index[client_index]] = -1; + + netcode_encryption_manager_remove_encryption_mapping( &server->encryption_manager, &server->client_address[client_index], server->time ); + + server->client_connected[client_index] = 0; + server->client_confirmed[client_index] = 0; + server->client_id[client_index] = 0; + server->client_sequence[client_index] = 0; + server->client_last_packet_send_time[client_index] = 0.0; + server->client_last_packet_receive_time[client_index] = 0.0; + memset( &server->client_address[client_index], 0, sizeof( struct netcode_address_t ) ); + server->client_encryption_index[client_index] = -1; + memset( server->client_user_data[client_index], 0, NETCODE_USER_DATA_BYTES ); + + server->num_connected_clients--; + + netcode_assert( server->num_connected_clients >= 0 ); +} + +void netcode_server_disconnect_client( struct netcode_server_t * server, int client_index ) +{ + netcode_assert( server ); + + if ( !server->running ) + return; + + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + netcode_assert( server->client_loopback[client_index] == 0 ); + + if ( !server->client_connected[client_index] ) + return; + + if ( server->client_loopback[client_index] ) + return; + + netcode_server_disconnect_client_internal( server, client_index, 1 ); +} + +void netcode_server_disconnect_all_clients( struct netcode_server_t * server ) +{ + netcode_assert( server ); + + if ( !server->running ) + return; + + int i; + for ( i = 0; i < server->max_clients; ++i ) + { + if ( server->client_connected[i] && !server->client_loopback[i] ) + { + netcode_server_disconnect_client_internal( server, i, 1 ); + } + } +} + +void netcode_server_stop( struct netcode_server_t * server ) +{ + netcode_assert( server ); + + if ( !server->running ) + return; + + netcode_server_disconnect_all_clients( server ); + + server->running = 0; + server->max_clients = 0; + server->num_connected_clients = 0; + + server->global_sequence = 0; + server->challenge_sequence = 0; + memset( server->challenge_key, 0, NETCODE_KEY_BYTES ); + + netcode_connect_token_entries_reset( server->connect_token_entries ); + + netcode_encryption_manager_reset( &server->encryption_manager ); + + netcode_printf( NETCODE_LOG_LEVEL_INFO, "server stopped\n" ); +} + +int netcode_server_find_client_index_by_id( struct netcode_server_t * server, uint64_t client_id ) +{ + netcode_assert( server ); + + int i; + for ( i = 0; i < server->max_clients; ++i ) + { + if ( server->client_connected[i] && server->client_id[i] == client_id ) + return i; + } + + return -1; +} + +int netcode_server_find_client_index_by_address( struct netcode_server_t * server, struct netcode_address_t * address ) +{ + netcode_assert( server ); + netcode_assert( address ); + + if ( address->type == 0 ) + return -1; + + int i; + for ( i = 0; i < server->max_clients; ++i ) + { + if ( server->client_connected[i] && netcode_address_equal( &server->client_address[i], address ) ) + return i; + } + + return -1; +} + +void netcode_server_process_connection_request_packet( struct netcode_server_t * server, + struct netcode_address_t * from, + struct netcode_connection_request_packet_t * packet ) +{ + netcode_assert( server ); + + (void) from; + + struct netcode_connect_token_private_t connect_token_private; + if ( netcode_read_connect_token_private( packet->connect_token_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES, &connect_token_private ) != NETCODE_OK ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection request. failed to read connect token\n" ); + return; + } + + int found_server_address = 0; + int i; + for ( i = 0; i < connect_token_private.num_server_addresses; ++i ) + { + if ( netcode_address_equal( &server->address, &connect_token_private.server_addresses[i] ) ) + { + found_server_address = 1; + } + } + if ( !found_server_address ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection request. server address not in connect token whitelist\n" ); + return; + } + + if ( netcode_server_find_client_index_by_address( server, from ) != -1 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection request. a client with this address is already connected\n" ); + return; + } + + if ( netcode_server_find_client_index_by_id( server, connect_token_private.client_id ) != -1 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection request. a client with this id is already connected\n" ); + return; + } + + if ( !netcode_connect_token_entries_find_or_add( server->connect_token_entries, + from, + packet->connect_token_data + NETCODE_CONNECT_TOKEN_PRIVATE_BYTES - NETCODE_MAC_BYTES, + server->time ) ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection request. connect token has already been used\n" ); + return; + } + + if ( server->num_connected_clients == server->max_clients ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server denied connection request. server is full\n" ); + + struct netcode_connection_denied_packet_t p; + p.packet_type = NETCODE_CONNECTION_DENIED_PACKET; + + netcode_server_send_global_packet( server, &p, from, connect_token_private.server_to_client_key ); + + return; + } + + double expire_time = ( connect_token_private.timeout_seconds >= 0 ) ? server->time + connect_token_private.timeout_seconds : -1.0; + + if ( !netcode_encryption_manager_add_encryption_mapping( &server->encryption_manager, + from, + connect_token_private.server_to_client_key, + connect_token_private.client_to_server_key, + server->time, + expire_time, + connect_token_private.timeout_seconds ) ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection request. failed to add encryption mapping\n" ); + return; + } + + struct netcode_challenge_token_t challenge_token; + challenge_token.client_id = connect_token_private.client_id; + memcpy( challenge_token.user_data, connect_token_private.user_data, NETCODE_USER_DATA_BYTES ); + + struct netcode_connection_challenge_packet_t challenge_packet; + challenge_packet.packet_type = NETCODE_CONNECTION_CHALLENGE_PACKET; + challenge_packet.challenge_token_sequence = server->challenge_sequence; + netcode_write_challenge_token( &challenge_token, challenge_packet.challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES ); + if ( netcode_encrypt_challenge_token( challenge_packet.challenge_token_data, + NETCODE_CHALLENGE_TOKEN_BYTES, + server->challenge_sequence, + server->challenge_key ) != NETCODE_OK ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection request. failed to encrypt challenge token\n" ); + return; + } + + server->challenge_sequence++; + + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server sent connection challenge packet\n" ); + + netcode_server_send_global_packet( server, &challenge_packet, from, connect_token_private.server_to_client_key ); +} + +int netcode_server_find_free_client_index( struct netcode_server_t * server ) +{ + netcode_assert( server ); + + int i; + for ( i = 0; i < server->max_clients; ++i ) + { + if ( !server->client_connected[i] ) + return i; + } + + return -1; +} + +void netcode_server_connect_client( struct netcode_server_t * server, + int client_index, + struct netcode_address_t * address, + uint64_t client_id, + int encryption_index, + int timeout_seconds, + void * user_data ) +{ + netcode_assert( server ); + netcode_assert( server->running ); + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + netcode_assert( address ); + netcode_assert( encryption_index != -1 ); + netcode_assert( user_data ); + netcode_assert( server->encryption_manager.client_index[encryption_index] == -1 ); + + server->num_connected_clients++; + + netcode_assert( server->num_connected_clients <= server->max_clients ); + + netcode_assert( server->client_connected[client_index] == 0 ); + + netcode_encryption_manager_set_expire_time( &server->encryption_manager, encryption_index, -1.0 ); + + server->encryption_manager.client_index[encryption_index] = client_index; + + server->client_connected[client_index] = 1; + server->client_timeout[client_index] = timeout_seconds; + server->client_encryption_index[client_index] = encryption_index; + server->client_id[client_index] = client_id; + server->client_sequence[client_index] = 0; + server->client_address[client_index] = *address; + server->client_last_packet_send_time[client_index] = server->time; + server->client_last_packet_receive_time[client_index] = server->time; + memcpy( server->client_user_data[client_index], user_data, NETCODE_USER_DATA_BYTES ); + + char address_string[NETCODE_MAX_ADDRESS_STRING_LENGTH]; + + netcode_printf( NETCODE_LOG_LEVEL_INFO, "server accepted client %s %.16" PRIx64 " in slot %d\n", + netcode_address_to_string( address, address_string ), client_id, client_index ); + + struct netcode_connection_keep_alive_packet_t packet; + packet.packet_type = NETCODE_CONNECTION_KEEP_ALIVE_PACKET; + packet.client_index = client_index; + packet.max_clients = server->max_clients; + + netcode_server_send_client_packet( server, &packet, client_index ); + + if ( server->config.connect_disconnect_callback ) + { + server->config.connect_disconnect_callback( server->config.callback_context, client_index, 1 ); + } +} + +void netcode_server_process_connection_response_packet( struct netcode_server_t * server, + struct netcode_address_t * from, + struct netcode_connection_response_packet_t * packet, + int encryption_index ) +{ + netcode_assert( server ); + + if ( netcode_decrypt_challenge_token( packet->challenge_token_data, + NETCODE_CHALLENGE_TOKEN_BYTES, + packet->challenge_token_sequence, + server->challenge_key ) != NETCODE_OK ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection response. failed to decrypt challenge token\n" ); + return; + } + + struct netcode_challenge_token_t challenge_token; + if ( netcode_read_challenge_token( packet->challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES, &challenge_token ) != NETCODE_OK ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection response. failed to read challenge token\n" ); + return; + } + + uint8_t * packet_send_key = netcode_encryption_manager_get_send_key( &server->encryption_manager, encryption_index ); + + if ( !packet_send_key ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection response. no packet send key\n" ); + return; + } + + if ( netcode_server_find_client_index_by_address( server, from ) != -1 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection response. a client with this address is already connected\n" ); + return; + } + + if ( netcode_server_find_client_index_by_id( server, challenge_token.client_id ) != -1 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server ignored connection response. a client with this id is already connected\n" ); + return; + } + + if ( server->num_connected_clients == server->max_clients ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server denied connection response. server is full\n" ); + + struct netcode_connection_denied_packet_t p; + p.packet_type = NETCODE_CONNECTION_DENIED_PACKET; + + netcode_server_send_global_packet( server, &p, from, packet_send_key ); + + return; + } + + int client_index = netcode_server_find_free_client_index( server ); + + netcode_assert( client_index != -1 ); + + int timeout_seconds = netcode_encryption_manager_get_timeout( &server->encryption_manager, encryption_index ); + + netcode_server_connect_client( server, client_index, from, challenge_token.client_id, encryption_index, timeout_seconds, challenge_token.user_data ); +} + +void netcode_server_process_packet_internal( struct netcode_server_t * server, + struct netcode_address_t * from, + void * packet, + uint64_t sequence, + int encryption_index, + int client_index ) +{ + netcode_assert( server ); + netcode_assert( packet ); + + (void) from; + (void) sequence; + + uint8_t packet_type = ( (uint8_t*) packet ) [0]; + + switch ( packet_type ) + { + case NETCODE_CONNECTION_REQUEST_PACKET: + { + if ( ( server->flags & NETCODE_SERVER_FLAG_IGNORE_CONNECTION_REQUEST_PACKETS ) == 0 ) + { + char from_address_string[NETCODE_MAX_ADDRESS_STRING_LENGTH]; + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server received connection request from %s\n", netcode_address_to_string( from, from_address_string ) ); + netcode_server_process_connection_request_packet( server, from, (struct netcode_connection_request_packet_t*) packet ); + } + } + break; + + case NETCODE_CONNECTION_RESPONSE_PACKET: + { + if ( ( server->flags & NETCODE_SERVER_FLAG_IGNORE_CONNECTION_RESPONSE_PACKETS ) == 0 ) + { + char from_address_string[NETCODE_MAX_ADDRESS_STRING_LENGTH]; + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server received connection response from %s\n", netcode_address_to_string( from, from_address_string ) ); + netcode_server_process_connection_response_packet( server, from, (struct netcode_connection_response_packet_t*) packet, encryption_index ); + } + } + break; + + case NETCODE_CONNECTION_KEEP_ALIVE_PACKET: + { + if ( client_index != -1 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server received connection keep alive packet from client %d\n", client_index ); + server->client_last_packet_receive_time[client_index] = server->time; + if ( !server->client_confirmed[client_index] ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server confirmed connection with client %d\n", client_index ); + server->client_confirmed[client_index] = 1; + } + } + } + break; + + case NETCODE_CONNECTION_PAYLOAD_PACKET: + { + if ( client_index != -1 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server received connection payload packet from client %d\n", client_index ); + server->client_last_packet_receive_time[client_index] = server->time; + if ( !server->client_confirmed[client_index] ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server confirmed connection with client %d\n", client_index ); + server->client_confirmed[client_index] = 1; + } + netcode_packet_queue_push( &server->client_packet_queue[client_index], packet, sequence ); + return; + } + } + break; + + case NETCODE_CONNECTION_DISCONNECT_PACKET: + { + if ( client_index != -1 ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server received disconnect packet from client %d\n", client_index ); + netcode_server_disconnect_client_internal( server, client_index, 0 ); + } + } + break; + + default: + break; + } + + server->config.free_function( server->config.allocator_context, packet ); +} + +void netcode_server_process_packet( struct netcode_server_t * server, struct netcode_address_t * from, uint8_t * packet_data, int packet_bytes ) +{ + uint8_t allowed_packets[NETCODE_CONNECTION_NUM_PACKETS]; + memset( allowed_packets, 0, sizeof( allowed_packets ) ); + allowed_packets[NETCODE_CONNECTION_REQUEST_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_RESPONSE_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_KEEP_ALIVE_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_PAYLOAD_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_DISCONNECT_PACKET] = 1; + + uint64_t current_timestamp = (uint64_t) time( NULL ); + + uint64_t sequence; + + int encryption_index = -1; + int client_index = netcode_server_find_client_index_by_address( server, from ); + if ( client_index != -1 ) + { + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + encryption_index = server->client_encryption_index[client_index]; + } + else + { + encryption_index = netcode_encryption_manager_find_encryption_mapping( &server->encryption_manager, from, server->time ); + } + + uint8_t * read_packet_key = netcode_encryption_manager_get_receive_key( &server->encryption_manager, encryption_index ); + + if ( !read_packet_key && packet_data[0] != 0 ) + { + char address_string[NETCODE_MAX_ADDRESS_STRING_LENGTH]; + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server could not process packet because no encryption mapping exists for %s\n", netcode_address_to_string( from, address_string ) ); + return; + } + + void * packet = netcode_read_packet( packet_data, + packet_bytes, + &sequence, + read_packet_key, + server->config.protocol_id, + current_timestamp, + server->config.private_key, + allowed_packets, + ( client_index != -1 ) ? &server->client_replay_protection[client_index] : NULL, + server->config.allocator_context, + server->config.allocate_function ); + + if ( !packet ) + return; + + netcode_server_process_packet_internal( server, from, packet, sequence, encryption_index, client_index ); +} + +void netcode_server_read_and_process_packet( struct netcode_server_t * server, + struct netcode_address_t * from, + uint8_t * packet_data, + int packet_bytes, + uint64_t current_timestamp, + uint8_t * allowed_packets ) +{ + if ( !server->running ) + return; + + if ( packet_bytes <= 1 ) + return; + + uint64_t sequence; + + int encryption_index = -1; + int client_index = netcode_server_find_client_index_by_address( server, from ); + if ( client_index != -1 ) + { + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + encryption_index = server->client_encryption_index[client_index]; + } + else + { + encryption_index = netcode_encryption_manager_find_encryption_mapping( &server->encryption_manager, from, server->time ); + } + + uint8_t * read_packet_key = netcode_encryption_manager_get_receive_key( &server->encryption_manager, encryption_index ); + + if ( !read_packet_key && packet_data[0] != 0 ) + { + char address_string[NETCODE_MAX_ADDRESS_STRING_LENGTH]; + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server could not process packet because no encryption mapping exists for %s\n", netcode_address_to_string( from, address_string ) ); + return; + } + + void * packet = netcode_read_packet( packet_data, + packet_bytes, + &sequence, + read_packet_key, + server->config.protocol_id, + current_timestamp, + server->config.private_key, + allowed_packets, + ( client_index != -1 ) ? &server->client_replay_protection[client_index] : NULL, + server->config.allocator_context, + server->config.allocate_function ); + + if ( !packet ) + return; + + netcode_server_process_packet_internal( server, from, packet, sequence, encryption_index, client_index ); +} + +void netcode_server_receive_packets( struct netcode_server_t * server ) +{ + netcode_assert( server ); + + uint8_t allowed_packets[NETCODE_CONNECTION_NUM_PACKETS]; + memset( allowed_packets, 0, sizeof( allowed_packets ) ); + allowed_packets[NETCODE_CONNECTION_REQUEST_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_RESPONSE_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_KEEP_ALIVE_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_PAYLOAD_PACKET] = 1; + allowed_packets[NETCODE_CONNECTION_DISCONNECT_PACKET] = 1; + + uint64_t current_timestamp = (uint64_t) time( NULL ); + + if ( !server->config.network_simulator ) + { + // process packets received from socket + + while ( 1 ) + { + struct netcode_address_t from; + + uint8_t packet_data[NETCODE_MAX_PACKET_BYTES]; + + int packet_bytes = 0; + + if ( server->config.override_send_and_receive ) + { + packet_bytes = server->config.receive_packet_override( server->config.callback_context, &from, packet_data, NETCODE_MAX_PACKET_BYTES ); + } + else + { + if (server->socket_holder.ipv4.handle != 0) + packet_bytes = netcode_socket_receive_packet( &server->socket_holder.ipv4, &from, packet_data, NETCODE_MAX_PACKET_BYTES ); + + if ( packet_bytes == 0 && server->socket_holder.ipv6.handle != 0) + packet_bytes = netcode_socket_receive_packet( &server->socket_holder.ipv6, &from, packet_data, NETCODE_MAX_PACKET_BYTES ); + } + + if ( packet_bytes == 0 ) + break; + + netcode_server_read_and_process_packet( server, &from, packet_data, packet_bytes, current_timestamp, allowed_packets ); + } + } + else + { + // process packets received from network simulator + + int num_packets_received = netcode_network_simulator_receive_packets( server->config.network_simulator, + &server->address, + NETCODE_SERVER_MAX_RECEIVE_PACKETS, + server->receive_packet_data, + server->receive_packet_bytes, + server->receive_from ); + + int i; + for ( i = 0; i < num_packets_received; ++i ) + { + netcode_server_read_and_process_packet( server, + &server->receive_from[i], + server->receive_packet_data[i], + server->receive_packet_bytes[i], + current_timestamp, + allowed_packets ); + + server->config.free_function( server->config.allocator_context, server->receive_packet_data[i] ); + } + } +} + +void netcode_server_send_packets( struct netcode_server_t * server ) +{ + netcode_assert( server ); + + if ( !server->running ) + return; + + int i; + for ( i = 0; i < server->max_clients; ++i ) + { + if ( server->client_connected[i] && !server->client_loopback[i] && + ( server->client_last_packet_send_time[i] + ( 1.0 / NETCODE_PACKET_SEND_RATE ) <= server->time ) ) + { + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server sent connection keep alive packet to client %d\n", i ); + struct netcode_connection_keep_alive_packet_t packet; + packet.packet_type = NETCODE_CONNECTION_KEEP_ALIVE_PACKET; + packet.client_index = i; + packet.max_clients = server->max_clients; + netcode_server_send_client_packet( server, &packet, i ); + } + } +} + +void netcode_server_check_for_timeouts( struct netcode_server_t * server ) +{ + netcode_assert( server ); + + if ( !server->running ) + return; + + int i; + for ( i = 0; i < server->max_clients; ++i ) + { + if ( server->client_connected[i] && server->client_timeout[i] > 0 && !server->client_loopback[i] && + ( server->client_last_packet_receive_time[i] + server->client_timeout[i] <= server->time ) ) + { + netcode_printf( NETCODE_LOG_LEVEL_INFO, "server timed out client %d\n", i ); + netcode_server_disconnect_client_internal( server, i, 0 ); + } + } +} + +int netcode_server_client_connected( struct netcode_server_t * server, int client_index ) +{ + netcode_assert( server ); + + if ( !server->running ) + return 0; + + if ( client_index < 0 || client_index >= server->max_clients ) + return 0; + + return server->client_connected[client_index]; +} + +uint64_t netcode_server_client_id( struct netcode_server_t * server, int client_index ) +{ + netcode_assert( server ); + + if ( !server->running ) + return 0; + + if ( client_index < 0 || client_index >= server->max_clients ) + return 0; + + return server->client_id[client_index]; +} + +struct netcode_address_t * netcode_server_client_address( struct netcode_server_t * server, int client_index ) +{ + netcode_assert( server ); + + if (!server->running) + return NULL; + + if (client_index < 0 || client_index >= server->max_clients) + return NULL; + + return &server->client_address[client_index]; +} + +uint64_t netcode_server_next_packet_sequence( struct netcode_server_t * server, int client_index ) +{ + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + if ( !server->client_connected[client_index] ) + return 0; + return server->client_sequence[client_index]; +} + +void netcode_server_send_packet( struct netcode_server_t * server, int client_index, NETCODE_CONST uint8_t * packet_data, int packet_bytes ) +{ + netcode_assert( server ); + netcode_assert( packet_data ); + netcode_assert( packet_bytes >= 0 ); + netcode_assert( packet_bytes <= NETCODE_MAX_PACKET_SIZE ); + + if ( !server->running ) + return; + + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + if ( !server->client_connected[client_index] ) + return; + + if ( !server->client_loopback[client_index] ) + { + uint8_t buffer[NETCODE_MAX_PAYLOAD_BYTES*2]; + + struct netcode_connection_payload_packet_t * packet = (struct netcode_connection_payload_packet_t*) buffer; + + packet->packet_type = NETCODE_CONNECTION_PAYLOAD_PACKET; + packet->payload_bytes = packet_bytes; + memcpy( packet->payload_data, packet_data, packet_bytes ); + + if ( !server->client_confirmed[client_index] ) + { + struct netcode_connection_keep_alive_packet_t keep_alive_packet; + keep_alive_packet.packet_type = NETCODE_CONNECTION_KEEP_ALIVE_PACKET; + keep_alive_packet.client_index = client_index; + keep_alive_packet.max_clients = server->max_clients; + netcode_server_send_client_packet( server, &keep_alive_packet, client_index ); + } + + netcode_server_send_client_packet( server, packet, client_index ); + } + else + { + netcode_assert( server->config.send_loopback_packet_callback ); + + server->config.send_loopback_packet_callback( server->config.callback_context, + client_index, + packet_data, + packet_bytes, + server->client_sequence[client_index]++ ); + + server->client_last_packet_send_time[client_index] = server->time; + } +} + +uint8_t * netcode_server_receive_packet( struct netcode_server_t * server, int client_index, int * packet_bytes, uint64_t * packet_sequence ) +{ + netcode_assert( server ); + netcode_assert( packet_bytes ); + + if ( !server->running ) + return NULL; + + if ( !server->client_connected[client_index] ) + return NULL; + + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + + struct netcode_connection_payload_packet_t * packet = (struct netcode_connection_payload_packet_t*) + netcode_packet_queue_pop( &server->client_packet_queue[client_index], packet_sequence ); + + if ( packet ) + { + netcode_assert( packet->packet_type == NETCODE_CONNECTION_PAYLOAD_PACKET ); + *packet_bytes = packet->payload_bytes; + netcode_assert( *packet_bytes >= 0 ); + netcode_assert( *packet_bytes <= NETCODE_MAX_PAYLOAD_BYTES ); + return (uint8_t*) &packet->payload_data; + } + else + { + return NULL; + } +} + +void netcode_server_free_packet( struct netcode_server_t * server, void * packet ) +{ + netcode_assert( server ); + netcode_assert( packet ); + (void) server; + int offset = offsetof( struct netcode_connection_payload_packet_t, payload_data ); + server->config.free_function( server->config.allocator_context, ( (uint8_t*) packet ) - offset ); +} + +int netcode_server_num_connected_clients( struct netcode_server_t * server ) +{ + netcode_assert( server ); + return server->num_connected_clients; +} + +void * netcode_server_client_user_data( struct netcode_server_t * server, int client_index ) +{ + netcode_assert( server ); + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + return server->client_user_data[client_index]; +} + +int netcode_server_running( struct netcode_server_t * server ) +{ + netcode_assert( server ); + return server->running; +} + +int netcode_server_max_clients( struct netcode_server_t * server ) +{ + return server->max_clients; +} + +void netcode_server_update( struct netcode_server_t * server, double time ) +{ + netcode_assert( server ); + server->time = time; + netcode_server_receive_packets( server ); + netcode_server_send_packets( server ); + netcode_server_check_for_timeouts( server ); +} + +void netcode_server_connect_loopback_client( struct netcode_server_t * server, int client_index, uint64_t client_id, NETCODE_CONST uint8_t * user_data ) +{ + netcode_assert( server ); + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + netcode_assert( server->running ); + netcode_assert( !server->client_connected[client_index] ); + + server->num_connected_clients++; + + netcode_assert( server->num_connected_clients <= server->max_clients ); + + server->client_loopback[client_index] = 1; + server->client_connected[client_index] = 1; + server->client_confirmed[client_index] = 1; + server->client_encryption_index[client_index] = -1; + server->client_id[client_index] = client_id; + server->client_sequence[client_index] = 0; + memset( &server->client_address[client_index], 0, sizeof( struct netcode_address_t ) ); + server->client_last_packet_send_time[client_index] = server->time; + server->client_last_packet_receive_time[client_index] = server->time; + + if ( user_data ) + { + memcpy( server->client_user_data[client_index], user_data, NETCODE_USER_DATA_BYTES ); + } + else + { + memset( server->client_user_data[client_index], 0, NETCODE_USER_DATA_BYTES ); + } + + netcode_printf( NETCODE_LOG_LEVEL_INFO, "server connected loopback client %.16" PRIx64 " in slot %d\n", client_id, client_index ); + + if ( server->config.connect_disconnect_callback ) + { + server->config.connect_disconnect_callback( server->config.callback_context, client_index, 1 ); + } +} + +void netcode_server_disconnect_loopback_client( struct netcode_server_t * server, int client_index ) +{ + netcode_assert( server ); + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + netcode_assert( server->running ); + netcode_assert( server->client_connected[client_index] ); + netcode_assert( server->client_loopback[client_index] ); + + netcode_printf( NETCODE_LOG_LEVEL_INFO, "server disconnected loopback client %d\n", client_index ); + + if ( server->config.connect_disconnect_callback ) + { + server->config.connect_disconnect_callback( server->config.callback_context, client_index, 0 ); + } + + while ( 1 ) + { + void * packet = netcode_packet_queue_pop( &server->client_packet_queue[client_index], NULL ); + if ( !packet ) + break; + server->config.free_function( server->config.allocator_context, packet ); + } + + netcode_packet_queue_clear( &server->client_packet_queue[client_index] ); + + server->client_connected[client_index] = 0; + server->client_loopback[client_index] = 0; + server->client_confirmed[client_index] = 0; + server->client_id[client_index] = 0; + server->client_sequence[client_index] = 0; + server->client_last_packet_send_time[client_index] = 0.0; + server->client_last_packet_receive_time[client_index] = 0.0; + memset( &server->client_address[client_index], 0, sizeof( struct netcode_address_t ) ); + server->client_encryption_index[client_index] = -1; + memset( server->client_user_data[client_index], 0, NETCODE_USER_DATA_BYTES ); + + server->num_connected_clients--; + + netcode_assert( server->num_connected_clients >= 0 ); +} + +int netcode_server_client_loopback( struct netcode_server_t * server, int client_index ) +{ + netcode_assert( server ); + netcode_assert( server->running ); + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + return server->client_loopback[client_index]; +} + +void netcode_server_process_loopback_packet( struct netcode_server_t * server, int client_index, NETCODE_CONST uint8_t * packet_data, int packet_bytes, uint64_t packet_sequence ) +{ + netcode_assert( server ); + netcode_assert( client_index >= 0 ); + netcode_assert( client_index < server->max_clients ); + netcode_assert( packet_data ); + netcode_assert( packet_bytes >= 0 ); + netcode_assert( packet_bytes <= NETCODE_MAX_PACKET_SIZE ); + netcode_assert( server->client_connected[client_index] ); + netcode_assert( server->client_loopback[client_index] ); + netcode_assert( server->running ); + + struct netcode_connection_payload_packet_t * packet = netcode_create_payload_packet( packet_bytes, server->config.allocator_context, server->config.allocate_function ); + if ( !packet ) + return; + + memcpy( packet->payload_data, packet_data, packet_bytes ); + + netcode_printf( NETCODE_LOG_LEVEL_DEBUG, "server processing loopback packet from client %d\n", client_index ); + + server->client_last_packet_receive_time[client_index] = server->time; + + netcode_packet_queue_push( &server->client_packet_queue[client_index], packet, packet_sequence ); +} + +uint16_t netcode_server_get_port( struct netcode_server_t * server ) +{ + netcode_assert( server ); + return server->address.type == NETCODE_ADDRESS_IPV4 ? server->socket_holder.ipv4.address.port : server->socket_holder.ipv6.address.port; +} + +// ---------------------------------------------------------------- + +int netcode_generate_connect_token( int num_server_addresses, + NETCODE_CONST char ** public_server_addresses, + NETCODE_CONST char ** internal_server_addresses, + int expire_seconds, + int timeout_seconds, + uint64_t client_id, + uint64_t protocol_id, + NETCODE_CONST uint8_t * private_key, + uint8_t * user_data, + uint8_t * output_buffer ) +{ + netcode_assert( num_server_addresses > 0 ); + netcode_assert( num_server_addresses <= NETCODE_MAX_SERVERS_PER_CONNECT ); + netcode_assert( public_server_addresses ); + netcode_assert( internal_server_addresses ); + netcode_assert( private_key ); + netcode_assert( user_data ); + netcode_assert( output_buffer ); + + // parse public server addresses + + struct netcode_address_t parsed_public_server_addresses[NETCODE_MAX_SERVERS_PER_CONNECT]; + int i; + for ( i = 0; i < num_server_addresses; ++i ) + { + if ( netcode_parse_address( public_server_addresses[i], &parsed_public_server_addresses[i] ) != NETCODE_OK ) + { + return NETCODE_ERROR; + } + } + + // parse internal server addresses + + struct netcode_address_t parsed_internal_server_addresses[NETCODE_MAX_SERVERS_PER_CONNECT]; + for ( i = 0; i < num_server_addresses; ++i ) + { + if ( netcode_parse_address( internal_server_addresses[i], &parsed_internal_server_addresses[i] ) != NETCODE_OK ) + { + return NETCODE_ERROR; + } + } + + // generate a connect token + + uint8_t nonce[NETCODE_CONNECT_TOKEN_NONCE_BYTES]; + netcode_generate_nonce(nonce); + + struct netcode_connect_token_private_t connect_token_private; + netcode_generate_connect_token_private( &connect_token_private, client_id, timeout_seconds, num_server_addresses, parsed_internal_server_addresses, user_data ); + + // write it to a buffer + + uint8_t connect_token_data[NETCODE_CONNECT_TOKEN_PRIVATE_BYTES]; + netcode_write_connect_token_private( &connect_token_private, connect_token_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + // encrypt the buffer + + uint64_t create_timestamp = time( NULL ); + uint64_t expire_timestamp = ( expire_seconds >= 0 ) ? ( create_timestamp + expire_seconds ) : 0xFFFFFFFFFFFFFFFFULL; + if ( netcode_encrypt_connect_token_private( connect_token_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES, NETCODE_VERSION_INFO, protocol_id, expire_timestamp, nonce, private_key ) != NETCODE_OK ) + return NETCODE_ERROR; + + // wrap a connect token around the private connect token data + + struct netcode_connect_token_t connect_token; + memcpy( connect_token.version_info, NETCODE_VERSION_INFO, NETCODE_VERSION_INFO_BYTES ); + connect_token.protocol_id = protocol_id; + connect_token.create_timestamp = create_timestamp; + connect_token.expire_timestamp = expire_timestamp; + memcpy( connect_token.nonce, nonce, NETCODE_CONNECT_TOKEN_NONCE_BYTES ); + memcpy( connect_token.private_data, connect_token_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + connect_token.num_server_addresses = num_server_addresses; + for ( i = 0; i < num_server_addresses; ++i ) + connect_token.server_addresses[i] = parsed_public_server_addresses[i]; + memcpy( connect_token.client_to_server_key, connect_token_private.client_to_server_key, NETCODE_KEY_BYTES ); + memcpy( connect_token.server_to_client_key, connect_token_private.server_to_client_key, NETCODE_KEY_BYTES ); + connect_token.timeout_seconds = timeout_seconds; + + // write the connect token to the output buffer + + netcode_write_connect_token( &connect_token, output_buffer, NETCODE_CONNECT_TOKEN_BYTES ); + + return NETCODE_OK; +} + +// --------------------------------------------------------------- + +#if __APPLE__ + +// MacOS + +#include +#include +#include + +void netcode_sleep( double time ) +{ + struct timespec ts; + ts.tv_sec = (time_t) time; + ts.tv_nsec = (long) ((time - (double) ( ts.tv_sec )) * 1000000000.0); + nanosleep( &ts, NULL ); +} + +static uint64_t start = 0; +static mach_timebase_info_data_t timebase_info; + +double netcode_time() +{ + if ( start == 0 ) + { + mach_timebase_info( &timebase_info ); + start = mach_absolute_time(); + return 0.0; + } + uint64_t current = mach_absolute_time(); + return ( (double) ( current - start ) ) * ( (double) timebase_info.numer ) / ( (double) timebase_info.denom ) / 1000000000.0; +} + +#elif __linux + +// linux + +#include + +void netcode_sleep( double time ) +{ + struct timespec ts; + ts.tv_sec = (time_t) time; + ts.tv_nsec = (long) ((time - (double) ( ts.tv_sec )) * 1000000000.0); + nanosleep( &ts, NULL ); +} + +double netcode_time() +{ + static double start = -1; + if ( start == -1 ) + { + struct timespec ts; + clock_gettime( CLOCK_MONOTONIC_RAW, &ts ); + start = ts.tv_sec + ( (double) ( ts.tv_nsec ) ) / 1000000000.0; + return 0.0; + } + struct timespec ts; + clock_gettime( CLOCK_MONOTONIC_RAW, &ts ); + double current = ts.tv_sec + ( (double) ( ts.tv_nsec ) ) / 1000000000.0; + return current - start; +} + +#elif defined( _WIN32 ) + +// windows + +#define NOMINMAX +#include + +void netcode_sleep( double time ) +{ + int milliseconds = (int) ( time * 1000 ); + Sleep( milliseconds ); +} + +static int timer_initialized = 0; +static LARGE_INTEGER timer_frequency; +static LARGE_INTEGER timer_start; + +double netcode_time() +{ + if ( !timer_initialized ) + { + QueryPerformanceFrequency( &timer_frequency ); + QueryPerformanceCounter( &timer_start ); + timer_initialized = 1; + } + LARGE_INTEGER now; + QueryPerformanceCounter( &now ); + return ( (double) ( now.QuadPart - timer_start.QuadPart ) ) / ( (double) ( timer_frequency.QuadPart ) ); +} + +#else + +#error unsupported platform! + +#endif + +// --------------------------------------------------------------- + +#if NETCODE_ENABLE_TESTS + +#include +#include +#include +#include + +static void check_handler( NETCODE_CONST char * condition, + NETCODE_CONST char * function, + NETCODE_CONST char * file, + int line ) +{ + printf( "check failed: ( %s ), function %s, file %s, line %d\n", condition, function, file, line ); +#ifdef NETCODE_DEBUG + #if defined( __GNUC__ ) + __builtin_trap(); + #elif defined( _MSC_VER ) + __debugbreak(); + #endif +#endif + exit( 1 ); +} + +#define check( condition ) \ +do \ +{ \ + if ( !(condition) ) \ + { \ + check_handler( #condition, (NETCODE_CONST char*) __FUNCTION__, (char*) __FILE__, __LINE__ ); \ + } \ +} while(0) + +static void test_queue() +{ + struct netcode_packet_queue_t queue; + + netcode_packet_queue_init( &queue, NULL, NULL, NULL ); + + check( queue.num_packets == 0 ); + check( queue.start_index == 0 ); + + // attempting to pop a packet off an empty queue should return NULL + + check( netcode_packet_queue_pop( &queue, NULL ) == NULL ); + + // add some packets to the queue and make sure they pop off in the correct order + { + #define NUM_PACKETS 100 + + void * packets[NUM_PACKETS]; + + int i; + for ( i = 0; i < NUM_PACKETS; ++i ) + { + packets[i] = malloc( (i+1) * 256 ); + check( netcode_packet_queue_push( &queue, packets[i], (uint64_t) i ) == 1 ); + } + + check( queue.num_packets == NUM_PACKETS ); + + for ( i = 0; i < NUM_PACKETS; ++i ) + { + uint64_t sequence = 0; + void * packet = netcode_packet_queue_pop( &queue, &sequence ); + check( sequence == (uint64_t) i ) ; + check( packet == packets[i] ); + free( packet ); + } + } + + // after all entries are popped off, the queue is empty, so calls to pop should return NULL + + check( queue.num_packets == 0 ); + + check( netcode_packet_queue_pop( &queue, NULL ) == NULL ); + + // test that the packet queue can be filled to max capacity + + void * packets[NETCODE_PACKET_QUEUE_SIZE]; + + int i; + for ( i = 0; i < NETCODE_PACKET_QUEUE_SIZE; ++i ) + { + packets[i] = malloc( i * 256 ); + check( netcode_packet_queue_push( &queue, packets[i], (uint64_t) i ) == 1 ); + } + + check( queue.num_packets == NETCODE_PACKET_QUEUE_SIZE ); + + // when the queue is full, attempting to push a packet should fail and return 0 + + check( netcode_packet_queue_push( &queue, malloc( 100 ), 0 ) == 0 ); + + // make sure all packets pop off in the correct order + + for ( i = 0; i < NETCODE_PACKET_QUEUE_SIZE; ++i ) + { + uint64_t sequence = 0; + void * packet = netcode_packet_queue_pop( &queue, &sequence ); + check( sequence == (uint64_t) i ); + check( packet == packets[i] ); + free( packet ); + } + + // add some packets again + + for ( i = 0; i < NETCODE_PACKET_QUEUE_SIZE; ++i ) + { + packets[i] = malloc( i * 256 ); + check( netcode_packet_queue_push( &queue, packets[i], (uint64_t) i ) == 1 ); + } + + // clear the queue and make sure that all packets are freed + + netcode_packet_queue_clear( &queue ); + + check( queue.start_index == 0 ); + check( queue.num_packets == 0 ); + for ( i = 0; i < NETCODE_PACKET_QUEUE_SIZE; ++i ) + check( queue.packet_data[i] == NULL ); +} + +static void test_endian() +{ + uint32_t value = 0x11223344; + + char * bytes = (char*) &value; + +#if NETCODE_LITTLE_ENDIAN + + check( bytes[0] == 0x44 ); + check( bytes[1] == 0x33 ); + check( bytes[2] == 0x22 ); + check( bytes[3] == 0x11 ); + +#else // #if NETCODE_LITTLE_ENDIAN + + check( bytes[3] == 0x44 ); + check( bytes[2] == 0x33 ); + check( bytes[1] == 0x22 ); + check( bytes[0] == 0x11 ); + +#endif // #if NETCODE_LITTLE_ENDIAN +} + +static void test_sequence() +{ + check( netcode_sequence_number_bytes_required( 0 ) == 1 ); + check( netcode_sequence_number_bytes_required( 0x11 ) == 1 ); + check( netcode_sequence_number_bytes_required( 0x1122 ) == 2 ); + check( netcode_sequence_number_bytes_required( 0x112233 ) == 3 ); + check( netcode_sequence_number_bytes_required( 0x11223344 ) == 4 ); + check( netcode_sequence_number_bytes_required( 0x1122334455 ) == 5 ); + check( netcode_sequence_number_bytes_required( 0x112233445566 ) == 6 ); + check( netcode_sequence_number_bytes_required( 0x11223344556677 ) == 7 ); + check( netcode_sequence_number_bytes_required( 0x1122334455667788 ) == 8 ); +} + +static void test_address() +{ + { + struct netcode_address_t address; + check( netcode_parse_address( "", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "[", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "[]", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "[]:", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( ":", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "1", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "12", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "123", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "1234", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "1234.0.12313.0000", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "1234.0.12313.0000.0.0.0.0.0", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "1312313:123131:1312313:123131:1312313:123131:1312313:123131:1312313:123131:1312313:123131", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( ".", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "..", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "...", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( "....", &address ) == NETCODE_ERROR ); + check( netcode_parse_address( ".....", &address ) == NETCODE_ERROR ); + } + + { + struct netcode_address_t address; + check( netcode_parse_address( "107.77.207.77", &address ) == NETCODE_OK ); + check( address.type == NETCODE_ADDRESS_IPV4 ); + check( address.port == 0 ); + check( address.data.ipv4[0] == 107 ); + check( address.data.ipv4[1] == 77 ); + check( address.data.ipv4[2] == 207 ); + check( address.data.ipv4[3] == 77 ); + } + + { + struct netcode_address_t address; + check( netcode_parse_address( "127.0.0.1", &address ) == NETCODE_OK ); + check( address.type == NETCODE_ADDRESS_IPV4 ); + check( address.port == 0 ); + check( address.data.ipv4[0] == 127 ); + check( address.data.ipv4[1] == 0 ); + check( address.data.ipv4[2] == 0 ); + check( address.data.ipv4[3] == 1 ); + } + + { + struct netcode_address_t address; + check( netcode_parse_address( "107.77.207.77:40000", &address ) == NETCODE_OK ); + check( address.type == NETCODE_ADDRESS_IPV4 ); + check( address.port == 40000 ); + check( address.data.ipv4[0] == 107 ); + check( address.data.ipv4[1] == 77 ); + check( address.data.ipv4[2] == 207 ); + check( address.data.ipv4[3] == 77 ); + } + + { + struct netcode_address_t address; + check( netcode_parse_address( "127.0.0.1:40000", &address ) == NETCODE_OK ); + check( address.type == NETCODE_ADDRESS_IPV4 ); + check( address.port == 40000 ); + check( address.data.ipv4[0] == 127 ); + check( address.data.ipv4[1] == 0 ); + check( address.data.ipv4[2] == 0 ); + check( address.data.ipv4[3] == 1 ); + } + + { + struct netcode_address_t address; + check( netcode_parse_address( "fe80::202:b3ff:fe1e:8329", &address ) == NETCODE_OK ); + check( address.type == NETCODE_ADDRESS_IPV6 ); + check( address.port == 0 ); + check( address.data.ipv6[0] == 0xfe80 ); + check( address.data.ipv6[1] == 0x0000 ); + check( address.data.ipv6[2] == 0x0000 ); + check( address.data.ipv6[3] == 0x0000 ); + check( address.data.ipv6[4] == 0x0202 ); + check( address.data.ipv6[5] == 0xb3ff ); + check( address.data.ipv6[6] == 0xfe1e ); + check( address.data.ipv6[7] == 0x8329 ); + } + + { + struct netcode_address_t address; + check( netcode_parse_address( "::", &address ) == NETCODE_OK ); + check( address.type == NETCODE_ADDRESS_IPV6 ); + check( address.port == 0 ); + check( address.data.ipv6[0] == 0x0000 ); + check( address.data.ipv6[1] == 0x0000 ); + check( address.data.ipv6[2] == 0x0000 ); + check( address.data.ipv6[3] == 0x0000 ); + check( address.data.ipv6[4] == 0x0000 ); + check( address.data.ipv6[5] == 0x0000 ); + check( address.data.ipv6[6] == 0x0000 ); + check( address.data.ipv6[7] == 0x0000 ); + } + + { + struct netcode_address_t address; + check( netcode_parse_address( "::1", &address ) == NETCODE_OK ); + check( address.type == NETCODE_ADDRESS_IPV6 ); + check( address.port == 0 ); + check( address.data.ipv6[0] == 0x0000 ); + check( address.data.ipv6[1] == 0x0000 ); + check( address.data.ipv6[2] == 0x0000 ); + check( address.data.ipv6[3] == 0x0000 ); + check( address.data.ipv6[4] == 0x0000 ); + check( address.data.ipv6[5] == 0x0000 ); + check( address.data.ipv6[6] == 0x0000 ); + check( address.data.ipv6[7] == 0x0001 ); + } + + { + struct netcode_address_t address; + check( netcode_parse_address( "[fe80::202:b3ff:fe1e:8329]:40000", &address ) == NETCODE_OK ); + check( address.type == NETCODE_ADDRESS_IPV6 ); + check( address.port == 40000 ); + check( address.data.ipv6[0] == 0xfe80 ); + check( address.data.ipv6[1] == 0x0000 ); + check( address.data.ipv6[2] == 0x0000 ); + check( address.data.ipv6[3] == 0x0000 ); + check( address.data.ipv6[4] == 0x0202 ); + check( address.data.ipv6[5] == 0xb3ff ); + check( address.data.ipv6[6] == 0xfe1e ); + check( address.data.ipv6[7] == 0x8329 ); + } + + { + struct netcode_address_t address; + check( netcode_parse_address( "[::]:40000", &address ) == NETCODE_OK ); + check( address.type == NETCODE_ADDRESS_IPV6 ); + check( address.port == 40000 ); + check( address.data.ipv6[0] == 0x0000 ); + check( address.data.ipv6[1] == 0x0000 ); + check( address.data.ipv6[2] == 0x0000 ); + check( address.data.ipv6[3] == 0x0000 ); + check( address.data.ipv6[4] == 0x0000 ); + check( address.data.ipv6[5] == 0x0000 ); + check( address.data.ipv6[6] == 0x0000 ); + check( address.data.ipv6[7] == 0x0000 ); + } + + { + struct netcode_address_t address; + check( netcode_parse_address( "[::1]:40000", &address ) == NETCODE_OK ); + check( address.type == NETCODE_ADDRESS_IPV6 ); + check( address.port == 40000 ); + check( address.data.ipv6[0] == 0x0000 ); + check( address.data.ipv6[1] == 0x0000 ); + check( address.data.ipv6[2] == 0x0000 ); + check( address.data.ipv6[3] == 0x0000 ); + check( address.data.ipv6[4] == 0x0000 ); + check( address.data.ipv6[5] == 0x0000 ); + check( address.data.ipv6[6] == 0x0000 ); + check( address.data.ipv6[7] == 0x0001 ); + } +} + +#define TEST_PROTOCOL_ID 0x1122334455667788ULL +#define TEST_CLIENT_ID 0x1ULL +#define TEST_SERVER_PORT 40000 +#define TEST_CONNECT_TOKEN_EXPIRY 30 +#define TEST_TIMEOUT_SECONDS 15 + +static void test_connect_token() +{ + // generate a connect token + + struct netcode_address_t server_address; + server_address.type = NETCODE_ADDRESS_IPV4; + server_address.data.ipv4[0] = 127; + server_address.data.ipv4[1] = 0; + server_address.data.ipv4[2] = 0; + server_address.data.ipv4[3] = 1; + server_address.port = TEST_SERVER_PORT; + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes( user_data, NETCODE_USER_DATA_BYTES ); + + struct netcode_connect_token_private_t input_token; + + netcode_generate_connect_token_private( &input_token, TEST_CLIENT_ID, TEST_TIMEOUT_SECONDS, 1, &server_address, user_data ); + + check( input_token.client_id == TEST_CLIENT_ID ); + check( input_token.num_server_addresses == 1 ); + check( memcmp( input_token.user_data, user_data, NETCODE_USER_DATA_BYTES ) == 0 ); + check( netcode_address_equal( &input_token.server_addresses[0], &server_address ) ); + + // write it to a buffer + + uint8_t buffer[NETCODE_CONNECT_TOKEN_PRIVATE_BYTES]; + + netcode_write_connect_token_private( &input_token, buffer, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + // encrypt the buffer + + uint64_t expire_timestamp = time( NULL ) + 30; + uint8_t nonce[NETCODE_CONNECT_TOKEN_NONCE_BYTES]; + netcode_generate_nonce(nonce); + uint8_t key[NETCODE_KEY_BYTES]; + netcode_generate_key( key ); + + check( netcode_encrypt_connect_token_private( buffer, + NETCODE_CONNECT_TOKEN_PRIVATE_BYTES, + NETCODE_VERSION_INFO, + TEST_PROTOCOL_ID, + expire_timestamp, + nonce, + key ) == NETCODE_OK ); + + // decrypt the buffer + + check( netcode_decrypt_connect_token_private( buffer, + NETCODE_CONNECT_TOKEN_PRIVATE_BYTES, + NETCODE_VERSION_INFO, + TEST_PROTOCOL_ID, + expire_timestamp, + nonce, + key ) == NETCODE_OK ); + + // read the connect token back in + + struct netcode_connect_token_private_t output_token; + + check( netcode_read_connect_token_private( buffer, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES, &output_token ) == NETCODE_OK ); + + // make sure that everything matches the original connect token + + check( output_token.client_id == input_token.client_id ); + check( output_token.timeout_seconds == input_token.timeout_seconds ); + check( output_token.num_server_addresses == input_token.num_server_addresses ); + check( netcode_address_equal( &output_token.server_addresses[0], &input_token.server_addresses[0] ) ); + check( memcmp( output_token.client_to_server_key, input_token.client_to_server_key, NETCODE_KEY_BYTES ) == 0 ); + check( memcmp( output_token.server_to_client_key, input_token.server_to_client_key, NETCODE_KEY_BYTES ) == 0 ); + check( memcmp( output_token.user_data, input_token.user_data, NETCODE_USER_DATA_BYTES ) == 0 ); +} + +static void test_challenge_token() +{ + // generate a challenge token + + struct netcode_challenge_token_t input_token; + + input_token.client_id = TEST_CLIENT_ID; + netcode_random_bytes( input_token.user_data, NETCODE_USER_DATA_BYTES ); + + // write it to a buffer + + uint8_t buffer[NETCODE_CHALLENGE_TOKEN_BYTES]; + + netcode_write_challenge_token( &input_token, buffer, NETCODE_CHALLENGE_TOKEN_BYTES ); + + // encrypt the buffer + + uint64_t sequence = 1000; + uint8_t key[NETCODE_KEY_BYTES]; + netcode_generate_key( key ); + + check( netcode_encrypt_challenge_token( buffer, NETCODE_CHALLENGE_TOKEN_BYTES, sequence, key ) == NETCODE_OK ); + + // decrypt the buffer + + check( netcode_decrypt_challenge_token( buffer, NETCODE_CHALLENGE_TOKEN_BYTES, sequence, key ) == NETCODE_OK ); + + // read the challenge token back in + + struct netcode_challenge_token_t output_token; + + check( netcode_read_challenge_token( buffer, NETCODE_CHALLENGE_TOKEN_BYTES, &output_token ) == NETCODE_OK ); + + // make sure that everything matches the original challenge token + + check( output_token.client_id == input_token.client_id ); + check( memcmp( output_token.user_data, input_token.user_data, NETCODE_USER_DATA_BYTES ) == 0 ); +} + +static void test_connection_request_packet() +{ + // generate a connect token + + struct netcode_address_t server_address; + server_address.type = NETCODE_ADDRESS_IPV4; + server_address.data.ipv4[0] = 127; + server_address.data.ipv4[1] = 0; + server_address.data.ipv4[2] = 0; + server_address.data.ipv4[3] = 1; + server_address.port = TEST_SERVER_PORT; + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes( user_data, NETCODE_USER_DATA_BYTES ); + + struct netcode_connect_token_private_t input_token; + + netcode_generate_connect_token_private( &input_token, TEST_CLIENT_ID, TEST_TIMEOUT_SECONDS, 1, &server_address, user_data ); + + check( input_token.client_id == TEST_CLIENT_ID ); + check( input_token.num_server_addresses == 1 ); + check( memcmp( input_token.user_data, user_data, NETCODE_USER_DATA_BYTES ) == 0 ); + check( netcode_address_equal( &input_token.server_addresses[0], &server_address ) ); + + // write the conect token to a buffer (non-encrypted) + + uint8_t connect_token_data[NETCODE_CONNECT_TOKEN_PRIVATE_BYTES]; + + netcode_write_connect_token_private( &input_token, connect_token_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + // copy to a second buffer then encrypt it in place (we need the unencrypted token for verification later on) + + uint8_t encrypted_connect_token_data[NETCODE_CONNECT_TOKEN_PRIVATE_BYTES]; + + memcpy( encrypted_connect_token_data, connect_token_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + uint64_t connect_token_expire_timestamp = time( NULL ) + 30; + uint8_t connect_token_nonce[NETCODE_CONNECT_TOKEN_NONCE_BYTES]; + netcode_generate_nonce(connect_token_nonce); + uint8_t connect_token_key[NETCODE_KEY_BYTES]; + netcode_generate_key( connect_token_key ); + + check( netcode_encrypt_connect_token_private( encrypted_connect_token_data, + NETCODE_CONNECT_TOKEN_PRIVATE_BYTES, + NETCODE_VERSION_INFO, + TEST_PROTOCOL_ID, + connect_token_expire_timestamp, + connect_token_nonce, + connect_token_key ) == NETCODE_OK ); + + // setup a connection request packet wrapping the encrypted connect token + + struct netcode_connection_request_packet_t input_packet; + + input_packet.packet_type = NETCODE_CONNECTION_REQUEST_PACKET; + memcpy( input_packet.version_info, NETCODE_VERSION_INFO, NETCODE_VERSION_INFO_BYTES ); + input_packet.protocol_id = TEST_PROTOCOL_ID; + input_packet.connect_token_expire_timestamp = connect_token_expire_timestamp; + memcpy( input_packet.connect_token_nonce, connect_token_nonce, NETCODE_CONNECT_TOKEN_NONCE_BYTES ); + memcpy( input_packet.connect_token_data, encrypted_connect_token_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + // write the connection request packet to a buffer + + uint8_t buffer[2048]; + + uint8_t packet_key[NETCODE_KEY_BYTES]; + + netcode_generate_key( packet_key ); + + int bytes_written = netcode_write_packet( &input_packet, buffer, sizeof( buffer ), 1000, packet_key, TEST_PROTOCOL_ID ); + + check( bytes_written > 0 ); + + // read the connection request packet back in from the buffer (the connect token data is decrypted as part of the read packet validation) + + uint64_t sequence = 1000; + + uint8_t allowed_packets[NETCODE_CONNECTION_NUM_PACKETS]; + memset( allowed_packets, 1, sizeof( allowed_packets ) ); + + struct netcode_connection_request_packet_t * output_packet = (struct netcode_connection_request_packet_t*) + netcode_read_packet( buffer, bytes_written, &sequence, packet_key, TEST_PROTOCOL_ID, time( NULL ), connect_token_key, allowed_packets, NULL, NULL, NULL ); + + check( output_packet ); + + // make sure the read packet matches what was written + + check( output_packet->packet_type == NETCODE_CONNECTION_REQUEST_PACKET ); + check( memcmp( output_packet->version_info, input_packet.version_info, NETCODE_VERSION_INFO_BYTES ) == 0 ); + check( output_packet->protocol_id == input_packet.protocol_id ); + check( output_packet->connect_token_expire_timestamp == input_packet.connect_token_expire_timestamp ); + check( memcmp( output_packet->connect_token_nonce, input_packet.connect_token_nonce, NETCODE_CONNECT_TOKEN_NONCE_BYTES ) == 0 ); + check( memcmp( output_packet->connect_token_data, connect_token_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES - NETCODE_MAC_BYTES ) == 0 ); + + free( output_packet ); +} + +void test_connection_denied_packet() +{ + // setup a connection denied packet + + struct netcode_connection_denied_packet_t input_packet; + + input_packet.packet_type = NETCODE_CONNECTION_DENIED_PACKET; + + // write the packet to a buffer + + uint8_t buffer[NETCODE_MAX_PACKET_BYTES]; + + uint8_t packet_key[NETCODE_KEY_BYTES]; + + netcode_generate_key( packet_key ); + + int bytes_written = netcode_write_packet( &input_packet, buffer, sizeof( buffer ), 1000, packet_key, TEST_PROTOCOL_ID ); + + check( bytes_written > 0 ); + + // read the packet back in from the buffer + + uint64_t sequence; + + uint8_t allowed_packet_types[NETCODE_CONNECTION_NUM_PACKETS]; + memset( allowed_packet_types, 1, sizeof( allowed_packet_types ) ); + + struct netcode_connection_denied_packet_t * output_packet = (struct netcode_connection_denied_packet_t*) + netcode_read_packet( buffer, bytes_written, &sequence, packet_key, TEST_PROTOCOL_ID, time( NULL ), NULL, allowed_packet_types, NULL, NULL, NULL ); + + check( output_packet ); + + // make sure the read packet matches what was written + + check( output_packet->packet_type == NETCODE_CONNECTION_DENIED_PACKET ); + + free( output_packet ); +} + +void test_connection_challenge_packet() +{ + // setup a connection challenge packet + + struct netcode_connection_challenge_packet_t input_packet; + + input_packet.packet_type = NETCODE_CONNECTION_CHALLENGE_PACKET; + input_packet.challenge_token_sequence = 0; + netcode_random_bytes( input_packet.challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES ); + + // write the packet to a buffer + + uint8_t buffer[NETCODE_MAX_PACKET_BYTES]; + + uint8_t packet_key[NETCODE_KEY_BYTES]; + + netcode_generate_key( packet_key ); + + int bytes_written = netcode_write_packet( &input_packet, buffer, sizeof( buffer ), 1000, packet_key, TEST_PROTOCOL_ID ); + + check( bytes_written > 0 ); + + // read the packet back in from the buffer + + uint64_t sequence; + + uint8_t allowed_packet_types[NETCODE_CONNECTION_NUM_PACKETS]; + memset( allowed_packet_types, 1, sizeof( allowed_packet_types ) ); + + struct netcode_connection_challenge_packet_t * output_packet = (struct netcode_connection_challenge_packet_t*) + netcode_read_packet( buffer, bytes_written, &sequence, packet_key, TEST_PROTOCOL_ID, time( NULL ), NULL, allowed_packet_types, NULL, NULL, NULL ); + + check( output_packet ); + + // make sure the read packet packet matches what was written + + check( output_packet->packet_type == NETCODE_CONNECTION_CHALLENGE_PACKET ); + check( output_packet->challenge_token_sequence == input_packet.challenge_token_sequence ); + check( memcmp( output_packet->challenge_token_data, input_packet.challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES ) == 0 ); + + free( output_packet ); +} + +void test_connection_response_packet() +{ + // setup a connection response packet + + struct netcode_connection_response_packet_t input_packet; + + input_packet.packet_type = NETCODE_CONNECTION_RESPONSE_PACKET; + input_packet.challenge_token_sequence = 0; + netcode_random_bytes( input_packet.challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES ); + + // write the packet to a buffer + + uint8_t buffer[NETCODE_MAX_PACKET_BYTES]; + + uint8_t packet_key[NETCODE_KEY_BYTES]; + + netcode_generate_key( packet_key ); + + int bytes_written = netcode_write_packet( &input_packet, buffer, sizeof( buffer ), 1000, packet_key, TEST_PROTOCOL_ID ); + + check( bytes_written > 0 ); + + // read the packet back in from the buffer + + uint64_t sequence; + + uint8_t allowed_packet_types[NETCODE_CONNECTION_NUM_PACKETS]; + memset( allowed_packet_types, 1, sizeof( allowed_packet_types ) ); + + struct netcode_connection_response_packet_t * output_packet = (struct netcode_connection_response_packet_t*) + netcode_read_packet( buffer, bytes_written, &sequence, packet_key, TEST_PROTOCOL_ID, time( NULL ), NULL, allowed_packet_types, NULL, NULL, NULL ); + + check( output_packet ); + + // make sure the read packet matches what was written + + check( output_packet->packet_type == NETCODE_CONNECTION_RESPONSE_PACKET ); + check( output_packet->challenge_token_sequence == input_packet.challenge_token_sequence ); + check( memcmp( output_packet->challenge_token_data, input_packet.challenge_token_data, NETCODE_CHALLENGE_TOKEN_BYTES ) == 0 ); + + free( output_packet ); +} + +void test_connection_keep_alive_packet() +{ + // setup a connection keep alive packet + + struct netcode_connection_keep_alive_packet_t input_packet; + + input_packet.packet_type = NETCODE_CONNECTION_KEEP_ALIVE_PACKET; + input_packet.client_index = 10; + input_packet.max_clients = 16; + + // write the packet to a buffer + + uint8_t buffer[NETCODE_MAX_PACKET_BYTES]; + + uint8_t packet_key[NETCODE_KEY_BYTES]; + + netcode_generate_key( packet_key ); + + int bytes_written = netcode_write_packet( &input_packet, buffer, sizeof( buffer ), 1000, packet_key, TEST_PROTOCOL_ID ); + + check( bytes_written > 0 ); + + // read the packet back in from the buffer + + uint64_t sequence; + + uint8_t allowed_packet_types[NETCODE_CONNECTION_NUM_PACKETS]; + memset( allowed_packet_types, 1, sizeof( allowed_packet_types ) ); + + struct netcode_connection_keep_alive_packet_t * output_packet = (struct netcode_connection_keep_alive_packet_t*) + netcode_read_packet( buffer, bytes_written, &sequence, packet_key, TEST_PROTOCOL_ID, time( NULL ), NULL, allowed_packet_types, NULL, NULL, NULL ); + + check( output_packet ); + + // make sure the read packet matches what was written + + check( output_packet->packet_type == NETCODE_CONNECTION_KEEP_ALIVE_PACKET ); + check( output_packet->client_index == input_packet.client_index ); + check( output_packet->max_clients == input_packet.max_clients ); + + free( output_packet ); +} + +void test_connection_payload_packet() +{ + // setup a connection payload packet + + struct netcode_connection_payload_packet_t * input_packet = netcode_create_payload_packet( NETCODE_MAX_PAYLOAD_BYTES, NULL, NULL ); + + check( input_packet->packet_type == NETCODE_CONNECTION_PAYLOAD_PACKET ); + check( input_packet->payload_bytes == NETCODE_MAX_PAYLOAD_BYTES ); + + netcode_random_bytes( input_packet->payload_data, NETCODE_MAX_PAYLOAD_BYTES ); + + // write the packet to a buffer + + uint8_t buffer[NETCODE_MAX_PACKET_BYTES]; + + uint8_t packet_key[NETCODE_KEY_BYTES]; + + netcode_generate_key( packet_key ); + + int bytes_written = netcode_write_packet( input_packet, buffer, sizeof( buffer ), 1000, packet_key, TEST_PROTOCOL_ID ); + + check( bytes_written > 0 ); + + // read the packet back in from the buffer + + uint64_t sequence; + + uint8_t allowed_packet_types[NETCODE_CONNECTION_NUM_PACKETS]; + memset( allowed_packet_types, 1, sizeof( allowed_packet_types ) ); + + struct netcode_connection_payload_packet_t * output_packet = (struct netcode_connection_payload_packet_t*) + netcode_read_packet( buffer, bytes_written, &sequence, packet_key, TEST_PROTOCOL_ID, time( NULL ), NULL, allowed_packet_types, NULL, NULL, NULL ); + + check( output_packet ); + + // make sure the read packet matches what was written + + check( output_packet->packet_type == NETCODE_CONNECTION_PAYLOAD_PACKET ); + check( output_packet->payload_bytes == input_packet->payload_bytes ); + check( memcmp( output_packet->payload_data, input_packet->payload_data, NETCODE_MAX_PAYLOAD_BYTES ) == 0 ); + + free( input_packet ); + free( output_packet ); +} + +void test_connection_disconnect_packet() +{ + // setup a connection disconnect packet + + struct netcode_connection_disconnect_packet_t input_packet; + + input_packet.packet_type = NETCODE_CONNECTION_DISCONNECT_PACKET; + + // write the packet to a buffer + + uint8_t buffer[NETCODE_MAX_PACKET_BYTES]; + + uint8_t packet_key[NETCODE_KEY_BYTES]; + + netcode_generate_key( packet_key ); + + int bytes_written = netcode_write_packet( &input_packet, buffer, sizeof( buffer ), 1000, packet_key, TEST_PROTOCOL_ID ); + + check( bytes_written > 0 ); + + // read the packet back in from the buffer + + uint64_t sequence; + + uint8_t allowed_packet_types[NETCODE_CONNECTION_NUM_PACKETS]; + memset( allowed_packet_types, 1, sizeof( allowed_packet_types ) ); + + struct netcode_connection_disconnect_packet_t * output_packet = (struct netcode_connection_disconnect_packet_t*) + netcode_read_packet( buffer, bytes_written, &sequence, packet_key, TEST_PROTOCOL_ID, time( NULL ), NULL, allowed_packet_types, NULL, NULL, NULL ); + + check( output_packet ); + + // make sure the read packet matches what was written + + check( output_packet->packet_type == NETCODE_CONNECTION_DISCONNECT_PACKET ); + + free( output_packet ); +} + +void test_connect_token_public() +{ + // generate a private connect token + + struct netcode_address_t server_address; + server_address.type = NETCODE_ADDRESS_IPV4; + server_address.data.ipv4[0] = 127; + server_address.data.ipv4[1] = 0; + server_address.data.ipv4[2] = 0; + server_address.data.ipv4[3] = 1; + server_address.port = TEST_SERVER_PORT; + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes( user_data, NETCODE_USER_DATA_BYTES ); + + struct netcode_connect_token_private_t connect_token_private; + + netcode_generate_connect_token_private( &connect_token_private, TEST_CLIENT_ID, TEST_TIMEOUT_SECONDS, 1, &server_address, user_data ); + + check( connect_token_private.client_id == TEST_CLIENT_ID ); + check( connect_token_private.num_server_addresses == 1 ); + check( memcmp( connect_token_private.user_data, user_data, NETCODE_USER_DATA_BYTES ) == 0 ); + check( netcode_address_equal( &connect_token_private.server_addresses[0], &server_address ) ); + + // write it to a buffer + + uint8_t connect_token_private_data[NETCODE_CONNECT_TOKEN_PRIVATE_BYTES]; + netcode_write_connect_token_private( &connect_token_private, connect_token_private_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + + // encrypt the buffer + + uint64_t create_timestamp = time( NULL ); + uint64_t expire_timestamp = create_timestamp + 30; + uint8_t connect_token_nonce[NETCODE_CONNECT_TOKEN_NONCE_BYTES]; + netcode_generate_nonce( connect_token_nonce ); + uint8_t key[NETCODE_KEY_BYTES]; + netcode_generate_key( key ); + check( netcode_encrypt_connect_token_private( connect_token_private_data, + NETCODE_CONNECT_TOKEN_PRIVATE_BYTES, + NETCODE_VERSION_INFO, + TEST_PROTOCOL_ID, + expire_timestamp, + connect_token_nonce, + key ) == 1 ); + + // wrap a public connect token around the private connect token data + + struct netcode_connect_token_t input_connect_token; + memset( &input_connect_token, 0, sizeof( struct netcode_connect_token_t ) ); + memcpy( input_connect_token.version_info, NETCODE_VERSION_INFO, NETCODE_VERSION_INFO_BYTES ); + input_connect_token.protocol_id = TEST_PROTOCOL_ID; + input_connect_token.create_timestamp = create_timestamp; + input_connect_token.expire_timestamp = expire_timestamp; + memcpy( input_connect_token.nonce, connect_token_nonce, NETCODE_CONNECT_TOKEN_NONCE_BYTES ); + memcpy( input_connect_token.private_data, connect_token_private_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ); + input_connect_token.num_server_addresses = 1; + input_connect_token.server_addresses[0] = server_address; + memcpy( input_connect_token.client_to_server_key, connect_token_private.client_to_server_key, NETCODE_KEY_BYTES ); + memcpy( input_connect_token.server_to_client_key, connect_token_private.server_to_client_key, NETCODE_KEY_BYTES ); + input_connect_token.timeout_seconds = (int) TEST_TIMEOUT_SECONDS; + + // write the connect token to a buffer + + uint8_t buffer[NETCODE_CONNECT_TOKEN_BYTES]; + netcode_write_connect_token( &input_connect_token, buffer, NETCODE_CONNECT_TOKEN_BYTES ); + + // read the buffer back in + + struct netcode_connect_token_t output_connect_token; + memset( &output_connect_token, 0, sizeof( struct netcode_connect_token_t ) ); + check( netcode_read_connect_token( buffer, NETCODE_CONNECT_TOKEN_BYTES, &output_connect_token ) == 1 ); + + // make sure the public connect token matches what was written + + check( memcmp( output_connect_token.version_info, input_connect_token.version_info, NETCODE_VERSION_INFO_BYTES ) == 0 ); + check( output_connect_token.protocol_id == input_connect_token.protocol_id ); + check( output_connect_token.create_timestamp == input_connect_token.create_timestamp ); + check( output_connect_token.expire_timestamp == input_connect_token.expire_timestamp ); + check( memcmp( output_connect_token.nonce, input_connect_token.nonce, NETCODE_CONNECT_TOKEN_NONCE_BYTES ) == 0 ); + check( memcmp( output_connect_token.private_data, input_connect_token.private_data, NETCODE_CONNECT_TOKEN_PRIVATE_BYTES ) == 0 ); + check( output_connect_token.num_server_addresses == input_connect_token.num_server_addresses ); + check( netcode_address_equal( &output_connect_token.server_addresses[0], &input_connect_token.server_addresses[0] ) ); + check( memcmp( output_connect_token.client_to_server_key, input_connect_token.client_to_server_key, NETCODE_KEY_BYTES ) == 0 ); + check( memcmp( output_connect_token.server_to_client_key, input_connect_token.server_to_client_key, NETCODE_KEY_BYTES ) == 0 ); + check( output_connect_token.timeout_seconds == input_connect_token.timeout_seconds ); +} + +void test_encryption_manager() +{ + struct netcode_encryption_manager_t encryption_manager; + + netcode_encryption_manager_reset( &encryption_manager ); + + double time = 100.0; + + // generate some test encryption mappings + + struct encryption_mapping_t + { + struct netcode_address_t address; + uint8_t send_key[NETCODE_KEY_BYTES]; + uint8_t receive_key[NETCODE_KEY_BYTES]; + }; + + #define NUM_ENCRYPTION_MAPPINGS 5 + + struct encryption_mapping_t encryption_mapping[NUM_ENCRYPTION_MAPPINGS]; + memset( encryption_mapping, 0, sizeof( encryption_mapping ) ); + int i; + for ( i = 0; i < NUM_ENCRYPTION_MAPPINGS; ++i ) + { + encryption_mapping[i].address.type = NETCODE_ADDRESS_IPV6; + encryption_mapping[i].address.data.ipv6[7] = 1; + encryption_mapping[i].address.port = ( uint16_t) ( 20000 + i ); + netcode_generate_key( encryption_mapping[i].send_key ); + netcode_generate_key( encryption_mapping[i].receive_key ); + } + + // add the encryption mappings to the manager and make sure they can be looked up by address + + for ( i = 0; i < NUM_ENCRYPTION_MAPPINGS; ++i ) + { + int encryption_index = netcode_encryption_manager_find_encryption_mapping( &encryption_manager, &encryption_mapping[i].address, time ); + + check( encryption_index == -1 ); + + check( netcode_encryption_manager_get_send_key( &encryption_manager, encryption_index ) == NULL ); + check( netcode_encryption_manager_get_receive_key( &encryption_manager, encryption_index ) == NULL ); + + check( netcode_encryption_manager_add_encryption_mapping( &encryption_manager, + &encryption_mapping[i].address, + encryption_mapping[i].send_key, + encryption_mapping[i].receive_key, + time, + -1.0, + TEST_TIMEOUT_SECONDS ) ); + + encryption_index = netcode_encryption_manager_find_encryption_mapping( &encryption_manager, &encryption_mapping[i].address, time ); + + uint8_t * send_key = netcode_encryption_manager_get_send_key( &encryption_manager, encryption_index ); + uint8_t * receive_key = netcode_encryption_manager_get_receive_key( &encryption_manager, encryption_index ); + + check( send_key ); + check( receive_key ); + + check( memcmp( send_key, encryption_mapping[i].send_key, NETCODE_KEY_BYTES ) == 0 ); + check( memcmp( receive_key, encryption_mapping[i].receive_key, NETCODE_KEY_BYTES ) == 0 ); + } + + // removing an encryption mapping that doesn't exist should return 0 + { + struct netcode_address_t address; + address.type = NETCODE_ADDRESS_IPV6; + address.data.ipv6[7] = 1; + address.port = 50000; + + check( netcode_encryption_manager_remove_encryption_mapping( &encryption_manager, &address, time ) == 0 ); + } + + // remove the first and last encryption mappings + + check( netcode_encryption_manager_remove_encryption_mapping( &encryption_manager, &encryption_mapping[0].address, time ) == 1 ); + + check( netcode_encryption_manager_remove_encryption_mapping( &encryption_manager, &encryption_mapping[NUM_ENCRYPTION_MAPPINGS-1].address, time ) == 1 ); + + // make sure the encryption mappings that were removed can no longer be looked up by address + + for ( i = 0; i < NUM_ENCRYPTION_MAPPINGS; ++i ) + { + int encryption_index = netcode_encryption_manager_find_encryption_mapping( &encryption_manager, &encryption_mapping[i].address, time ); + + uint8_t * send_key = netcode_encryption_manager_get_send_key( &encryption_manager, encryption_index ); + uint8_t * receive_key = netcode_encryption_manager_get_receive_key( &encryption_manager, encryption_index ); + + if ( i != 0 && i != NUM_ENCRYPTION_MAPPINGS - 1 ) + { + check( send_key ); + check( receive_key ); + + check( memcmp( send_key, encryption_mapping[i].send_key, NETCODE_KEY_BYTES ) == 0 ); + check( memcmp( receive_key, encryption_mapping[i].receive_key, NETCODE_KEY_BYTES ) == 0 ); + } + else + { + check( !send_key ); + check( !receive_key ); + } + } + + // add the encryption mappings back in + + check( netcode_encryption_manager_add_encryption_mapping( &encryption_manager, + &encryption_mapping[0].address, + encryption_mapping[0].send_key, + encryption_mapping[0].receive_key, + time, + -1.0, + TEST_TIMEOUT_SECONDS ) ); + + check( netcode_encryption_manager_add_encryption_mapping( &encryption_manager, + &encryption_mapping[NUM_ENCRYPTION_MAPPINGS-1].address, + encryption_mapping[NUM_ENCRYPTION_MAPPINGS-1].send_key, + encryption_mapping[NUM_ENCRYPTION_MAPPINGS-1].receive_key, + time, + -1.0, + TEST_TIMEOUT_SECONDS ) ); + + // all encryption mappings should be able to be looked up by address again + + for ( i = 0; i < NUM_ENCRYPTION_MAPPINGS; ++i ) + { + int encryption_index = netcode_encryption_manager_find_encryption_mapping( &encryption_manager, &encryption_mapping[i].address, time ); + + uint8_t * send_key = netcode_encryption_manager_get_send_key( &encryption_manager, encryption_index ); + uint8_t * receive_key = netcode_encryption_manager_get_receive_key( &encryption_manager, encryption_index ); + + check( send_key ); + check( receive_key ); + + check( memcmp( send_key, encryption_mapping[i].send_key, NETCODE_KEY_BYTES ) == 0 ); + check( memcmp( receive_key, encryption_mapping[i].receive_key, NETCODE_KEY_BYTES ) == 0 ); + } + + // check that encryption mappings time out properly + + time += TEST_TIMEOUT_SECONDS * 2; + + for ( i = 0; i < NUM_ENCRYPTION_MAPPINGS; ++i ) + { + int encryption_index = netcode_encryption_manager_find_encryption_mapping( &encryption_manager, &encryption_mapping[i].address, time ); + + uint8_t * send_key = netcode_encryption_manager_get_send_key( &encryption_manager, encryption_index ); + uint8_t * receive_key = netcode_encryption_manager_get_receive_key( &encryption_manager, encryption_index ); + + check( !send_key ); + check( !receive_key ); + } + + // add the same encryption mappings after timeout + + for ( i = 0; i < NUM_ENCRYPTION_MAPPINGS; ++i ) + { + int encryption_index = netcode_encryption_manager_find_encryption_mapping( &encryption_manager, &encryption_mapping[i].address, time ); + + check( encryption_index == -1 ); + + check( netcode_encryption_manager_get_send_key( &encryption_manager, encryption_index ) == NULL ); + check( netcode_encryption_manager_get_receive_key( &encryption_manager, encryption_index ) == NULL ); + + check( netcode_encryption_manager_add_encryption_mapping( &encryption_manager, + &encryption_mapping[i].address, + encryption_mapping[i].send_key, + encryption_mapping[i].receive_key, + time, + -1.0, + TEST_TIMEOUT_SECONDS ) ); + + encryption_index = netcode_encryption_manager_find_encryption_mapping( &encryption_manager, &encryption_mapping[i].address, time ); + + uint8_t * send_key = netcode_encryption_manager_get_send_key( &encryption_manager, encryption_index ); + uint8_t * receive_key = netcode_encryption_manager_get_receive_key( &encryption_manager, encryption_index ); + + check( send_key ); + check( receive_key ); + + check( memcmp( send_key, encryption_mapping[i].send_key, NETCODE_KEY_BYTES ) == 0 ); + check( memcmp( receive_key, encryption_mapping[i].receive_key, NETCODE_KEY_BYTES ) == 0 ); + } + + // reset the encryption mapping and verify that all encryption mappings have been removed + + netcode_encryption_manager_reset( &encryption_manager ); + + for ( i = 0; i < NUM_ENCRYPTION_MAPPINGS; ++i ) + { + int encryption_index = netcode_encryption_manager_find_encryption_mapping( &encryption_manager, &encryption_mapping[i].address, time ); + + uint8_t * send_key = netcode_encryption_manager_get_send_key( &encryption_manager, encryption_index ); + uint8_t * receive_key = netcode_encryption_manager_get_receive_key( &encryption_manager, encryption_index ); + + check( !send_key ); + check( !receive_key ); + } + + // test the expire time for encryption mapping works as expected + + check( netcode_encryption_manager_add_encryption_mapping( &encryption_manager, + &encryption_mapping[0].address, + encryption_mapping[0].send_key, + encryption_mapping[0].receive_key, + time, + time + 1.0, + TEST_TIMEOUT_SECONDS ) ); + + int encryption_index = netcode_encryption_manager_find_encryption_mapping( &encryption_manager, &encryption_mapping[0].address, time ); + + check( encryption_index != -1 ); + + check( netcode_encryption_manager_find_encryption_mapping( &encryption_manager, &encryption_mapping[0].address, time + 1.1f ) == -1 ); + + netcode_encryption_manager_set_expire_time( &encryption_manager, encryption_index, -1.0 ); + + check( netcode_encryption_manager_find_encryption_mapping( &encryption_manager, &encryption_mapping[0].address, time ) == encryption_index ); +} + +void test_replay_protection() +{ + struct netcode_replay_protection_t replay_protection; + + int i; + for ( i = 0; i < 2; ++i ) + { + netcode_replay_protection_reset( &replay_protection ); + + check( replay_protection.most_recent_sequence == 0 ); + + // the first time we receive packets, they should not be already received + + #define MAX_SEQUENCE ( NETCODE_REPLAY_PROTECTION_BUFFER_SIZE * 4 ) + + uint64_t sequence; + for ( sequence = 0; sequence < MAX_SEQUENCE; ++sequence ) + { + check( netcode_replay_protection_already_received( &replay_protection, sequence ) == 0 ); + netcode_replay_protection_advance_sequence( &replay_protection, sequence ); + } + + // old packets outside buffer should be considered already received + + check( netcode_replay_protection_already_received( &replay_protection, 0 ) == 1 ); + + // packets received a second time should be flagged already received + + for ( sequence = MAX_SEQUENCE - 10; sequence < MAX_SEQUENCE; ++sequence ) + { + check( netcode_replay_protection_already_received( &replay_protection, sequence ) == 1 ); + } + + // jumping ahead to a much higher sequence should be considered not already received + + check( netcode_replay_protection_already_received( &replay_protection, MAX_SEQUENCE + NETCODE_REPLAY_PROTECTION_BUFFER_SIZE ) == 0 ); + + // old packets should be considered already received + + for ( sequence = 0; sequence < MAX_SEQUENCE; ++sequence ) + { + check( netcode_replay_protection_already_received( &replay_protection, sequence ) == 1 ); + } + } +} + +void test_client_create() +{ + { + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create( "127.0.0.1:40000", &client_config, 0.0 ); + + struct netcode_address_t test_address; + netcode_parse_address( "127.0.0.1:40000", &test_address ); + + check( client ); + check( client->socket_holder.ipv4.handle != 0 ); + check( client->socket_holder.ipv6.handle == 0 ); + check( netcode_address_equal( &client->address, &test_address ) ); + + netcode_client_destroy( client ); + } + + { + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, 0.0 ); + + struct netcode_address_t test_address; + netcode_parse_address( "[::]:50000", &test_address ); + + check( client ); + check( client->socket_holder.ipv4.handle == 0 ); + check( client->socket_holder.ipv6.handle != 0 ); + check( netcode_address_equal( &client->address, &test_address ) ); + + netcode_client_destroy( client ); + } + + { + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create_overload( "127.0.0.1:40000", "[::]:50000", &client_config, 0.0 ); + + struct netcode_address_t test_address; + netcode_parse_address( "127.0.0.1:40000", &test_address ); + + check( client ); + check( client->socket_holder.ipv4.handle != 0 ); + check( client->socket_holder.ipv6.handle != 0 ); + check( netcode_address_equal( &client->address, &test_address ) ); + + netcode_client_destroy( client ); + } + + { + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create_overload( "[::]:50000", "127.0.0.1:40000", &client_config, 0.0 ); + + struct netcode_address_t test_address; + netcode_parse_address( "[::]:50000", &test_address ); + + check( client ); + check( client->socket_holder.ipv4.handle != 0 ); + check( client->socket_holder.ipv6.handle != 0 ); + check( netcode_address_equal( &client->address, &test_address ) ); + + netcode_client_destroy( client ); + } +} + +void test_server_create() +{ + { + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + + struct netcode_server_t * server = netcode_server_create( "127.0.0.1:40000", &server_config, 0.0 ); + + struct netcode_address_t test_address; + netcode_parse_address( "127.0.0.1:40000", &test_address ); + + check( server ); + check( server->socket_holder.ipv4.handle != 0 ); + check( server->socket_holder.ipv6.handle == 0 ); + check( netcode_address_equal( &server->address, &test_address ) ); + + netcode_server_destroy( server ); + } + + { + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:50000", &server_config, 0.0 ); + + struct netcode_address_t test_address; + netcode_parse_address( "[::1]:50000", &test_address ); + + check( server ); + check( server->socket_holder.ipv4.handle == 0 ); + check( server->socket_holder.ipv6.handle != 0 ); + check( netcode_address_equal( &server->address, &test_address ) ); + + netcode_server_destroy( server ); + } + + { + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + + struct netcode_server_t * server = netcode_server_create_overload( "127.0.0.1:40000", "[::1]:50000", &server_config, 0.0 ); + + struct netcode_address_t test_address; + netcode_parse_address( "127.0.0.1:40000", &test_address ); + + check( server ); + check( server->socket_holder.ipv4.handle != 0 ); + check( server->socket_holder.ipv6.handle != 0 ); + check( netcode_address_equal( &server->address, &test_address ) ); + + netcode_server_destroy( server ); + } + + { + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + + struct netcode_server_t * server = netcode_server_create_overload( "[::1]:50000", "127.0.0.1:40000", &server_config, 0.0 ); + + struct netcode_address_t test_address; + netcode_parse_address( "[::1]:50000", &test_address ); + + check( server ); + check( server->socket_holder.ipv4.handle != 0 ); + check( server->socket_holder.ipv6.handle != 0 ); + check( netcode_address_equal( &server->address, &test_address ) ); + + netcode_server_destroy( server ); + } +} + +static uint8_t private_key[NETCODE_KEY_BYTES] = { 0x60, 0x6a, 0xbe, 0x6e, 0xc9, 0x19, 0x10, 0xea, + 0x9a, 0x65, 0x62, 0xf6, 0x6f, 0x2b, 0x30, 0xe4, + 0x43, 0x71, 0xd6, 0x2c, 0xd1, 0x99, 0x27, 0x26, + 0x6b, 0x3c, 0x60, 0xf4, 0xb7, 0x15, 0xab, 0xa1 }; + +void test_client_server_connect() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( client ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + int server_num_packets_received = 0; + int client_num_packets_received = 0; + + uint8_t packet_data[NETCODE_MAX_PACKET_SIZE]; + int i; + for ( i = 0; i < NETCODE_MAX_PACKET_SIZE; ++i ) + packet_data[i] = (uint8_t) i; + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + netcode_client_send_packet( client, packet_data, NETCODE_MAX_PACKET_SIZE ); + + netcode_server_send_packet( server, 0, packet_data, NETCODE_MAX_PACKET_SIZE ); + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + uint8_t * packet = netcode_client_receive_packet( client, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + client_num_packets_received++; + netcode_client_free_packet( client, packet ); + } + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + void * packet = netcode_server_receive_packet( server, 0, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + server_num_packets_received++; + netcode_server_free_packet( server, packet ); + } + + if ( client_num_packets_received >= 10 && server_num_packets_received >= 10 ) + { + if ( netcode_server_client_connected( server, 0 ) ) + { + netcode_server_disconnect_client( server, 0 ); + } + } + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + time += delta_time; + } + + check( client_num_packets_received >= 10 && server_num_packets_received >= 10 ); + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_client_server_ipv4_socket_connect() +{ + { + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create( "0.0.0.0:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "127.0.0.1:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "127.0.0.1:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + } + + { + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create( "0.0.0.0:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create_overload( "127.0.0.1:40000", "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "127.0.0.1:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + } + + { + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create_overload( "0.0.0.0:50000", "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "127.0.0.1:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "127.0.0.1:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + } + + { + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create_overload( "0.0.0.0:50000", "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create_overload( "127.0.0.1:40000", "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "127.0.0.1:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + } +} + +void test_client_server_ipv6_socket_connect() +{ + { + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + } + + { + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create_overload( "127.0.0.1:40000", "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + } + + { + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create_overload( "0.0.0.0:50000", "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + } + + { + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + + struct netcode_client_t * client = netcode_client_create_overload( "0.0.0.0:50000", "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create_overload( "127.0.0.1:40000", "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + } +} + +void test_client_server_keep_alive() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + // connect client to server + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( client ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + // pump the client and server long enough that they would timeout without keep alive packets + + int num_iterations = (int) ( 1.25f * TEST_TIMEOUT_SECONDS / delta_time ) + 1; + + int i; + for ( i = 0; i < num_iterations; ++i ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( client ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_client_server_multiple_clients() +{ + #define NUM_START_STOP_ITERATIONS 3 + + int max_clients[NUM_START_STOP_ITERATIONS] = { 2, 32, 5 }; + + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + int i; + for ( i = 0; i < NUM_START_STOP_ITERATIONS; ++i ) + { + // start the server with max # of clients for this iteration + + netcode_server_start( server, max_clients[i] ); + + // create # of client objects for this iteration and connect to server + + struct netcode_client_t ** client = (struct netcode_client_t **) malloc( sizeof( struct netcode_client_t* ) * max_clients[i] ); + + check( client ); + + int j; + for ( j = 0; j < max_clients[i]; ++j ) + { + char client_address[NETCODE_MAX_ADDRESS_STRING_LENGTH]; + snprintf( client_address, sizeof(client_address), "[::]:%d", 50000 + j ); + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + client[j] = netcode_client_create( client_address, &client_config, time ); + + check( client[j] ); + + uint64_t client_id = j; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, + &server_address, + &server_address, + TEST_CONNECT_TOKEN_EXPIRY, + TEST_TIMEOUT_SECONDS, + client_id, + TEST_PROTOCOL_ID, + private_key, + user_data, + connect_token ) ); + + netcode_client_connect( client[j], connect_token ); + } + + // make sure all clients can connect + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + for ( j = 0; j < max_clients[i]; ++j ) + { + netcode_client_update( client[j], time ); + } + + netcode_server_update( server, time ); + + int num_connected_clients = 0; + + for ( j = 0; j < max_clients[i]; ++j ) + { + if ( netcode_client_state( client[j] ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client[j] ) == NETCODE_CLIENT_STATE_CONNECTED ) + num_connected_clients++; + } + + if ( num_connected_clients == max_clients[i] ) + break; + + time += delta_time; + } + + check( netcode_server_num_connected_clients( server ) == max_clients[i] ); + + for ( j = 0; j < max_clients[i]; ++j ) + { + check( netcode_client_state( client[j] ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_server_client_connected( server, j ) == 1 ); + } + + // make sure all clients can exchange packets with the server + + int * server_num_packets_received = (int*) malloc( sizeof(int) * max_clients[i] ); + int * client_num_packets_received = (int*) malloc( sizeof(int) * max_clients[i] ); + + memset( server_num_packets_received, 0, sizeof(int) * max_clients[i] ); + memset( client_num_packets_received, 0, sizeof(int) * max_clients[i] ); + + uint8_t packet_data[NETCODE_MAX_PACKET_SIZE]; + for ( j = 0; j < NETCODE_MAX_PACKET_SIZE; ++j ) + packet_data[j] = (uint8_t) j; + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + for ( j = 0; j < max_clients[i]; ++j ) + { + netcode_client_update( client[j], time ); + } + + netcode_server_update( server, time ); + + for ( j = 0; j < max_clients[i]; ++j ) + { + netcode_client_send_packet( client[j], packet_data, NETCODE_MAX_PACKET_SIZE ); + } + + for ( j = 0; j < max_clients[i]; ++j ) + { + netcode_server_send_packet( server, j, packet_data, NETCODE_MAX_PACKET_SIZE ); + } + + for ( j = 0; j < max_clients[i]; ++j ) + { + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + uint8_t * packet = netcode_client_receive_packet( client[j], &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + client_num_packets_received[j]++; + netcode_client_free_packet( client[j], packet ); + } + } + + for ( j = 0; j < max_clients[i]; ++j ) + { + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + void * packet = netcode_server_receive_packet( server, j, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + server_num_packets_received[j]++; + netcode_server_free_packet( server, packet ); + } + } + + int num_clients_ready = 0; + + for ( j = 0; j < max_clients[i]; ++j ) + { + if ( client_num_packets_received[j] >= 1 && server_num_packets_received[j] >= 1 ) + { + num_clients_ready++; + } + } + + if ( num_clients_ready == max_clients[i] ) + break; + + for ( j = 0; j < max_clients[i]; ++j ) + { + if ( netcode_client_state( client[j] ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + } + + time += delta_time; + } + + int num_clients_ready = 0; + + for ( j = 0; j < max_clients[i]; ++j ) + { + if ( client_num_packets_received[j] >= 1 && server_num_packets_received[j] >= 1 ) + { + num_clients_ready++; + } + } + + check( num_clients_ready == max_clients[i] ); + + free( server_num_packets_received ); + free( client_num_packets_received ); + + netcode_network_simulator_reset( network_simulator ); + + for ( j = 0; j < max_clients[i]; ++j ) + { + netcode_client_destroy( client[j] ); + } + + free( client ); + + netcode_server_stop( server ); + } + + netcode_server_destroy( server ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_client_server_multiple_servers() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address[] = { "10.10.10.10:1000", "100.100.100.100:50000", "[::1]:40000" }; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 3, server_address, server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( client ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + int server_num_packets_received = 0; + int client_num_packets_received = 0; + + uint8_t packet_data[NETCODE_MAX_PACKET_SIZE]; + int i; + for ( i = 0; i < NETCODE_MAX_PACKET_SIZE; ++i ) + packet_data[i] = (uint8_t) i; + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + netcode_client_send_packet( client, packet_data, NETCODE_MAX_PACKET_SIZE ); + + netcode_server_send_packet( server, 0, packet_data, NETCODE_MAX_PACKET_SIZE ); + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + uint8_t * packet = netcode_client_receive_packet( client, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + client_num_packets_received++; + netcode_client_free_packet( client, packet ); + } + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + void * packet = netcode_server_receive_packet( server, 0, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + server_num_packets_received++; + netcode_server_free_packet( server, packet ); + } + + if ( client_num_packets_received >= 10 && server_num_packets_received >= 10 ) + { + if ( netcode_server_client_connected( server, 0 ) ) + { + netcode_server_disconnect_client( server, 0 ); + } + } + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + time += delta_time; + } + + check( client_num_packets_received >= 10 && server_num_packets_received >= 10 ); + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_client_error_connect_token_expired() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + double time = 0.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, 0, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + netcode_client_update( client, time ); + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECT_TOKEN_EXPIRED ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_client_error_invalid_connect_token() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + double time = 0.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + netcode_random_bytes( connect_token, NETCODE_CONNECT_TOKEN_BYTES ); + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + netcode_client_connect( client, connect_token ); + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_INVALID_CONNECT_TOKEN ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_client_error_connection_timed_out() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + // connect a client to the server + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( client ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + // now disable updating the server and verify that the client times out + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTION_TIMED_OUT ); + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_client_error_connection_response_timeout() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + server->flags = NETCODE_SERVER_FLAG_IGNORE_CONNECTION_RESPONSE_PACKETS; + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTION_RESPONSE_TIMED_OUT ); + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_client_error_connection_request_timeout() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + double time = 0.0; + double delta_time = 1.0 / 60.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + server->flags = NETCODE_SERVER_FLAG_IGNORE_CONNECTION_REQUEST_PACKETS; + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTION_REQUEST_TIMED_OUT ); + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_client_error_connection_denied() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + // start a server and connect one client + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( client ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + // now attempt to connect a second client. the connection should be denied. + + struct netcode_client_t * client2 = netcode_client_create( "[::]:50001", &client_config, time ); + + check( client2 ); + + uint8_t connect_token2[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id2 = 0; + netcode_random_bytes( (uint8_t*) &client_id2, 8 ); + + uint8_t user_data2[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data2, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id2, TEST_PROTOCOL_ID, private_key, user_data2, connect_token2 ) ); + + netcode_client_connect( client2, connect_token2 ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_client_update( client2, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client2 ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_state( client2 ) == NETCODE_CLIENT_STATE_CONNECTION_DENIED ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + + netcode_client_destroy( client2 ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_client_side_disconnect() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + // start a server and connect one client + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( client ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + // disconnect client side and verify that the server sees that client disconnect cleanly, rather than timing out. + + netcode_client_disconnect( client ); + + int i; + for ( i = 0; i < 10; ++i ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_server_client_connected( server, 0 ) == 0 ) + break; + + time += delta_time; + } + + check( netcode_server_client_connected( server, 0 ) == 0 ); + check( netcode_server_num_connected_clients( server ) == 0 ); + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_server_side_disconnect() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + // start a server and connect one client + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( client ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + // disconnect server side and verify that the client disconnects cleanly, rather than timing out. + + netcode_server_disconnect_client( server, 0 ); + + int i; + for ( i = 0; i < 10; ++i ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_DISCONNECTED ); + check( netcode_server_client_connected( server, 0 ) == 0 ); + check( netcode_server_num_connected_clients( server ) == 0 ); + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_client_reconnect() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + // start a server and connect one client + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( client ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + // disconnect client on the server-side and wait until client sees the disconnect + + netcode_network_simulator_reset( network_simulator ); + + netcode_server_disconnect_client( server, 0 ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_DISCONNECTED ); + check( netcode_server_client_connected( server, 0 ) == 0 ); + check( netcode_server_num_connected_clients( server ) == 0 ); + + // now reconnect the client and verify they connect + + netcode_network_simulator_reset( network_simulator ); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( client ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +struct test_loopback_context_t +{ + struct netcode_client_t * client; + struct netcode_server_t * server; + int num_loopback_packets_sent_to_client; + int num_loopback_packets_sent_to_server; +}; + +void client_send_loopback_packet_callback( void * _context, int client_index, NETCODE_CONST uint8_t * packet_data, int packet_bytes, uint64_t packet_sequence ) +{ + (void) packet_sequence; + check( _context ); + check( client_index == 0 ); + check( packet_data ); + check( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + int i; + for ( i = 0; i < packet_bytes; ++i ) + { + check( packet_data[i] == (uint8_t) i ); + } + struct test_loopback_context_t * context = (struct test_loopback_context_t*) _context; + context->num_loopback_packets_sent_to_server++; + netcode_server_process_loopback_packet( context->server, client_index, packet_data, packet_bytes, packet_sequence ); +} + +void server_send_loopback_packet_callback( void * _context, int client_index, NETCODE_CONST uint8_t * packet_data, int packet_bytes, uint64_t packet_sequence ) +{ + (void) packet_sequence; + check( _context ); + check( client_index == 0 ); + check( packet_data ); + check( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + int i; + for ( i = 0; i < packet_bytes; ++i ) + { + check( packet_data[i] == (uint8_t) i ); + } + struct test_loopback_context_t * context = (struct test_loopback_context_t*) _context; + context->num_loopback_packets_sent_to_client++; + netcode_client_process_loopback_packet( context->client, packet_data, packet_bytes, packet_sequence ); +} + +void test_disable_timeout() +{ + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.network_simulator = network_simulator; + + struct netcode_client_t * client = netcode_client_create( "[::]:50000", &client_config, time ); + + check( client ); + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + netcode_server_start( server, 1 ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, -1, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( client ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + int server_num_packets_received = 0; + int client_num_packets_received = 0; + + uint8_t packet_data[NETCODE_MAX_PACKET_SIZE]; + int i; + for ( i = 0; i < NETCODE_MAX_PACKET_SIZE; ++i ) + packet_data[i] = (uint8_t) i; + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( client, time ); + + netcode_server_update( server, time ); + + netcode_client_send_packet( client, packet_data, NETCODE_MAX_PACKET_SIZE ); + + netcode_server_send_packet( server, 0, packet_data, NETCODE_MAX_PACKET_SIZE ); + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + uint8_t * packet = netcode_client_receive_packet( client, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + client_num_packets_received++; + netcode_client_free_packet( client, packet ); + } + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + void * packet = netcode_server_receive_packet( server, 0, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + server_num_packets_received++; + netcode_server_free_packet( server, packet ); + } + + if ( client_num_packets_received >= 10 && server_num_packets_received >= 10 ) + { + if ( netcode_server_client_connected( server, 0 ) ) + { + netcode_server_disconnect_client( server, 0 ); + } + } + + if ( netcode_client_state( client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + time += 1000.0f; // normally this would timeout the client + } + + check( client_num_packets_received >= 10 && server_num_packets_received >= 10 ); + + netcode_server_destroy( server ); + + netcode_client_destroy( client ); + + netcode_network_simulator_destroy( network_simulator ); +} + +void test_loopback() +{ + struct test_loopback_context_t context; + memset( &context, 0, sizeof( context ) ); + + struct netcode_network_simulator_t * network_simulator = netcode_network_simulator_create( NULL, NULL, NULL ); + + network_simulator->latency_milliseconds = 250; + network_simulator->jitter_milliseconds = 250; + network_simulator->packet_loss_percent = 5; + network_simulator->duplicate_packet_percent = 10; + + double time = 0.0; + double delta_time = 1.0 / 10.0; + + // start the server + + struct netcode_server_config_t server_config; + netcode_default_server_config( &server_config ); + server_config.protocol_id = TEST_PROTOCOL_ID; + server_config.network_simulator = network_simulator; + server_config.callback_context = &context; + server_config.send_loopback_packet_callback = server_send_loopback_packet_callback; + memcpy( &server_config.private_key, private_key, NETCODE_KEY_BYTES ); + + struct netcode_server_t * server = netcode_server_create( "[::1]:40000", &server_config, time ); + + check( server ); + + int max_clients = 2; + + netcode_server_start( server, max_clients ); + + context.server = server; + + // connect a loopback client in slot 0 + + struct netcode_client_config_t client_config; + netcode_default_client_config( &client_config ); + client_config.callback_context = &context; + client_config.send_loopback_packet_callback = client_send_loopback_packet_callback; + client_config.network_simulator = network_simulator; + + struct netcode_client_t * loopback_client = netcode_client_create( "[::]:50000", &client_config, time ); + check( loopback_client ); + netcode_client_connect_loopback( loopback_client, 0, max_clients ); + context.client = loopback_client; + + check( netcode_client_index( loopback_client ) == 0 ); + check( netcode_client_loopback( loopback_client ) == 1 ); + check( netcode_client_max_clients( loopback_client ) == max_clients ); + check( netcode_client_state( loopback_client ) == NETCODE_CLIENT_STATE_CONNECTED ); + + uint64_t client_id = 0; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + netcode_server_connect_loopback_client( server, 0, client_id, NULL ); + + check( netcode_server_client_loopback( server, 0 ) == 1 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + // connect a regular client in the other slot + + struct netcode_client_t * regular_client = netcode_client_create( "[::]:50001", &client_config, time ); + + check( regular_client ); + + NETCODE_CONST char * server_address = "[::1]:40000"; + + uint8_t connect_token[NETCODE_CONNECT_TOKEN_BYTES]; + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + + uint8_t user_data[NETCODE_USER_DATA_BYTES]; + netcode_random_bytes(user_data, NETCODE_USER_DATA_BYTES); + + check( netcode_generate_connect_token( 1, &server_address, &server_address, TEST_CONNECT_TOKEN_EXPIRY, TEST_TIMEOUT_SECONDS, client_id, TEST_PROTOCOL_ID, private_key, user_data, connect_token ) ); + + netcode_client_connect( regular_client, connect_token ); + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( regular_client, time ); + + netcode_server_update( server, time ); + + if ( netcode_client_state( regular_client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + if ( netcode_client_state( regular_client ) == NETCODE_CLIENT_STATE_CONNECTED ) + break; + + time += delta_time; + } + + check( netcode_client_state( regular_client ) == NETCODE_CLIENT_STATE_CONNECTED ); + check( netcode_client_index( regular_client ) == 1 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_client_connected( server, 1 ) == 1 ); + check( netcode_server_client_loopback( server, 0 ) == 1 ); + check( netcode_server_client_loopback( server, 1 ) == 0 ); + check( netcode_server_num_connected_clients( server ) == 2 ); + + // test that we can exchange packets for the regular client and the loopback client + + int loopback_client_num_packets_received = 0; + int loopback_server_num_packets_received = 0; + int regular_server_num_packets_received = 0; + int regular_client_num_packets_received = 0; + + uint8_t packet_data[NETCODE_MAX_PACKET_SIZE]; + int i; + for ( i = 0; i < NETCODE_MAX_PACKET_SIZE; ++i ) + packet_data[i] = (uint8_t) i; + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( regular_client, time ); + + netcode_server_update( server, time ); + + netcode_client_send_packet( loopback_client, packet_data, NETCODE_MAX_PACKET_SIZE ); + + netcode_client_send_packet( regular_client, packet_data, NETCODE_MAX_PACKET_SIZE ); + + netcode_server_send_packet( server, 0, packet_data, NETCODE_MAX_PACKET_SIZE ); + + netcode_server_send_packet( server, 1, packet_data, NETCODE_MAX_PACKET_SIZE ); + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + uint8_t * packet = netcode_client_receive_packet( loopback_client, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + loopback_client_num_packets_received++; + netcode_client_free_packet( loopback_client, packet ); + } + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + uint8_t * packet = netcode_client_receive_packet( regular_client, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + regular_client_num_packets_received++; + netcode_client_free_packet( regular_client, packet ); + } + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + void * packet = netcode_server_receive_packet( server, 0, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + loopback_server_num_packets_received++; + netcode_server_free_packet( server, packet ); + } + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + void * packet = netcode_server_receive_packet( server, 1, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + regular_server_num_packets_received++; + netcode_server_free_packet( server, packet ); + } + + if ( loopback_client_num_packets_received >= 10 && loopback_server_num_packets_received >= 10 && + regular_client_num_packets_received >= 10 && regular_server_num_packets_received >= 10 ) + break; + + if ( netcode_client_state( regular_client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + time += delta_time; + } + + check( loopback_client_num_packets_received >= 10 ); + check( loopback_server_num_packets_received >= 10 ); + check( regular_client_num_packets_received >= 10 ); + check( regular_server_num_packets_received >= 10 ); + check( context.num_loopback_packets_sent_to_client >= 10 ); + check( context.num_loopback_packets_sent_to_server >= 10 ); + + // verify that we can disconnect the loopback client + + check( netcode_server_client_loopback( server, 0 ) == 1 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 2 ); + + netcode_server_disconnect_loopback_client( server, 0 ); + + check( netcode_server_client_loopback( server, 0 ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 0 ); + check( netcode_server_num_connected_clients( server ) == 1 ); + + netcode_client_disconnect_loopback( loopback_client ); + + check( netcode_client_state( loopback_client ) == NETCODE_CLIENT_STATE_DISCONNECTED ); + + // verify that we can reconnect the loopback client + + netcode_random_bytes( (uint8_t*) &client_id, 8 ); + netcode_server_connect_loopback_client( server, 0, client_id, NULL ); + + check( netcode_server_client_loopback( server, 0 ) == 1 ); + check( netcode_server_client_loopback( server, 1 ) == 0 ); + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_client_connected( server, 1 ) == 1 ); + check( netcode_server_num_connected_clients( server ) == 2 ); + + netcode_client_connect_loopback( loopback_client, 0, max_clients ); + + check( netcode_client_index( loopback_client ) == 0 ); + check( netcode_client_loopback( loopback_client ) == 1 ); + check( netcode_client_max_clients( loopback_client ) == max_clients ); + check( netcode_client_state( loopback_client ) == NETCODE_CLIENT_STATE_CONNECTED ); + + // verify that we can exchange packets for both regular and loopback client post reconnect + + loopback_server_num_packets_received = 0; + loopback_client_num_packets_received = 0; + regular_server_num_packets_received = 0; + regular_client_num_packets_received = 0; + context.num_loopback_packets_sent_to_client = 0; + context.num_loopback_packets_sent_to_server = 0; + + while ( 1 ) + { + netcode_network_simulator_update( network_simulator, time ); + + netcode_client_update( regular_client, time ); + + netcode_server_update( server, time ); + + netcode_client_send_packet( loopback_client, packet_data, NETCODE_MAX_PACKET_SIZE ); + + netcode_client_send_packet( regular_client, packet_data, NETCODE_MAX_PACKET_SIZE ); + + netcode_server_send_packet( server, 0, packet_data, NETCODE_MAX_PACKET_SIZE ); + + netcode_server_send_packet( server, 1, packet_data, NETCODE_MAX_PACKET_SIZE ); + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + uint8_t * packet = netcode_client_receive_packet( loopback_client, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + loopback_client_num_packets_received++; + netcode_client_free_packet( loopback_client, packet ); + } + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + uint8_t * packet = netcode_client_receive_packet( regular_client, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + regular_client_num_packets_received++; + netcode_client_free_packet( regular_client, packet ); + } + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + void * packet = netcode_server_receive_packet( server, 0, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + loopback_server_num_packets_received++; + netcode_server_free_packet( server, packet ); + } + + while ( 1 ) + { + int packet_bytes; + uint64_t packet_sequence; + void * packet = netcode_server_receive_packet( server, 1, &packet_bytes, &packet_sequence ); + if ( !packet ) + break; + (void) packet_sequence; + netcode_assert( packet_bytes == NETCODE_MAX_PACKET_SIZE ); + netcode_assert( memcmp( packet, packet_data, NETCODE_MAX_PACKET_SIZE ) == 0 ); + regular_server_num_packets_received++; + netcode_server_free_packet( server, packet ); + } + + if ( loopback_client_num_packets_received >= 10 && loopback_server_num_packets_received >= 10 && + regular_client_num_packets_received >= 10 && regular_server_num_packets_received >= 10 ) + break; + + if ( netcode_client_state( regular_client ) <= NETCODE_CLIENT_STATE_DISCONNECTED ) + break; + + time += delta_time; + } + + check( loopback_client_num_packets_received >= 10 ); + check( loopback_server_num_packets_received >= 10 ); + check( regular_client_num_packets_received >= 10 ); + check( regular_server_num_packets_received >= 10 ); + check( context.num_loopback_packets_sent_to_client >= 10 ); + check( context.num_loopback_packets_sent_to_server >= 10 ); + + // verify the regular client times out but loopback client doesn't + + time += 100000.0; + + netcode_server_update( server, time ); + + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_client_connected( server, 1 ) == 0 ); + + netcode_client_update( loopback_client, time ); + + check( netcode_client_state( loopback_client ) == NETCODE_CLIENT_STATE_CONNECTED ); + + // verify that disconnect all clients leaves loopback clients alone + + netcode_server_disconnect_all_clients( server ); + + check( netcode_server_client_connected( server, 0 ) == 1 ); + check( netcode_server_client_connected( server, 1 ) == 0 ); + check( netcode_server_client_loopback( server, 0 ) == 1 ); + + // clean up + + netcode_client_destroy( regular_client ); + + netcode_client_destroy( loopback_client ); + + netcode_server_destroy( server ); + + netcode_network_simulator_destroy( network_simulator ); +} + +#define RUN_TEST( test_function ) \ + do \ + { \ + printf( #test_function "\n" ); \ + test_function(); \ + } \ + while (0) + +void netcode_test() +{ + //while ( 1 ) + { + RUN_TEST( test_queue ); + RUN_TEST( test_endian ); + RUN_TEST( test_address ); + RUN_TEST( test_sequence ); + RUN_TEST( test_connect_token ); + RUN_TEST( test_challenge_token ); + RUN_TEST( test_connection_request_packet ); + RUN_TEST( test_connection_denied_packet ); + RUN_TEST( test_connection_challenge_packet ); + RUN_TEST( test_connection_response_packet ); + RUN_TEST( test_connection_payload_packet ); + RUN_TEST( test_connection_disconnect_packet ); + RUN_TEST( test_connect_token_public ); + RUN_TEST( test_encryption_manager ); + RUN_TEST( test_replay_protection ); + RUN_TEST( test_client_create ); + RUN_TEST( test_server_create ); + RUN_TEST( test_client_server_connect ); + RUN_TEST( test_client_server_ipv4_socket_connect ); + RUN_TEST( test_client_server_ipv6_socket_connect ); + RUN_TEST( test_client_server_keep_alive ); + RUN_TEST( test_client_server_multiple_clients ); + RUN_TEST( test_client_server_multiple_servers ); + RUN_TEST( test_client_error_connect_token_expired ); + RUN_TEST( test_client_error_invalid_connect_token ); + RUN_TEST( test_client_error_connection_timed_out ); + RUN_TEST( test_client_error_connection_response_timeout ); + RUN_TEST( test_client_error_connection_request_timeout ); + RUN_TEST( test_client_error_connection_denied ); + RUN_TEST( test_client_side_disconnect ); + RUN_TEST( test_server_side_disconnect ); + RUN_TEST( test_client_reconnect ); + RUN_TEST( test_disable_timeout ); + RUN_TEST( test_loopback ); + } +} + +#endif // #if NETCODE_ENABLE_TESTS + diff --git a/netcode/netcode.h b/netcode/netcode.h new file mode 100755 index 00000000..d60f3f00 --- /dev/null +++ b/netcode/netcode.h @@ -0,0 +1,293 @@ +/* + netcode reference implementation + + Copyright © 2017 - 2024, Mas Bandwidth LLC + + Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: + + 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the distribution. + + 3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived + from this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, + INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, + WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE + USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ + +#ifndef NETCODE_H +#define NETCODE_H + +#include + +#if !defined(NETCODE_DEBUG) && !defined(NETCODE_RELEASE) +#if defined(NDEBUG) +#define NETCODE_RELEASE +#else +#define NETCODE_DEBUG +#endif +#elif defined(NETCODE_DEBUG) && defined(NETCODE_RELEASE) +#error Can only define one of debug & release +#endif + +#if defined(__386__) || defined(i386) || defined(__i386__) \ + || defined(__X86) || defined(_M_IX86) \ + || defined(_M_X64) || defined(__x86_64__) \ + || defined(alpha) || defined(__alpha) || defined(__alpha__) \ + || defined(_M_ALPHA) \ + || defined(ARM) || defined(_ARM) || defined(__arm__) \ + || defined(__aarch64__) \ + || defined(WIN32) || defined(_WIN32) || defined(__WIN32__) \ + || defined(_WIN32_WCE) || defined(__NT__) \ + || defined(__MIPSEL__) + #define NETCODE_LITTLE_ENDIAN 1 +#else + #define NETCODE_BIG_ENDIAN 1 +#endif + +#define NETCODE_PLATFORM_WINDOWS 1 +#define NETCODE_PLATFORM_MAC 2 +#define NETCODE_PLATFORM_UNIX 3 + +#if defined(_WIN32) +#define NETCODE_PLATFORM NETCODE_PLATFORM_WINDOWS +#elif defined(__APPLE__) +#define NETCODE_PLATFORM NETCODE_PLATFORM_MAC +#else +#define NETCODE_PLATFORM NETCODE_PLATFORM_UNIX +#endif + +#define NETCODE_CONNECT_TOKEN_BYTES 2048 +#define NETCODE_KEY_BYTES 32 +#define NETCODE_MAC_BYTES 16 +#define NETCODE_USER_DATA_BYTES 256 +#define NETCODE_MAX_SERVERS_PER_CONNECT 32 + +#define NETCODE_CLIENT_STATE_CONNECT_TOKEN_EXPIRED -6 +#define NETCODE_CLIENT_STATE_INVALID_CONNECT_TOKEN -5 +#define NETCODE_CLIENT_STATE_CONNECTION_TIMED_OUT -4 +#define NETCODE_CLIENT_STATE_CONNECTION_RESPONSE_TIMED_OUT -3 +#define NETCODE_CLIENT_STATE_CONNECTION_REQUEST_TIMED_OUT -2 +#define NETCODE_CLIENT_STATE_CONNECTION_DENIED -1 +#define NETCODE_CLIENT_STATE_DISCONNECTED 0 +#define NETCODE_CLIENT_STATE_SENDING_CONNECTION_REQUEST 1 +#define NETCODE_CLIENT_STATE_SENDING_CONNECTION_RESPONSE 2 +#define NETCODE_CLIENT_STATE_CONNECTED 3 + +#define NETCODE_MAX_CLIENTS 256 +#define NETCODE_MAX_PACKET_SIZE 1200 + +#define NETCODE_LOG_LEVEL_NONE 0 +#define NETCODE_LOG_LEVEL_ERROR 1 +#define NETCODE_LOG_LEVEL_INFO 2 +#define NETCODE_LOG_LEVEL_DEBUG 3 + +#define NETCODE_OK 1 +#define NETCODE_ERROR 0 + +#define NETCODE_ADDRESS_NONE 0 +#define NETCODE_ADDRESS_IPV4 1 +#define NETCODE_ADDRESS_IPV6 2 + +#ifdef __cplusplus +#define NETCODE_CONST const +extern "C" { +#else +#if defined(__STDC__) +#define NETCODE_CONST const +#else +#define NETCODE_CONST +#endif +#endif + +int netcode_init(); + +void netcode_term(); + +struct netcode_address_t +{ + union { uint8_t ipv4[4]; uint16_t ipv6[8]; } data; + uint16_t port; + uint8_t type; +}; + +int netcode_parse_address( NETCODE_CONST char * address_string_in, struct netcode_address_t * address ); + +char * netcode_address_to_string( struct netcode_address_t * address, char * buffer ); + +int netcode_address_equal( struct netcode_address_t * a, struct netcode_address_t * b ); + +struct netcode_client_config_t +{ + void * allocator_context; + void * (*allocate_function)(void*,uint64_t); + void (*free_function)(void*,void*); + struct netcode_network_simulator_t * network_simulator; + void * callback_context; + void (*state_change_callback)(void*,int,int); + void (*send_loopback_packet_callback)(void*,int,NETCODE_CONST uint8_t*,int,uint64_t); + int override_send_and_receive; + void (*send_packet_override)(void*,struct netcode_address_t*,NETCODE_CONST uint8_t*,int); + int (*receive_packet_override)(void*,struct netcode_address_t*,uint8_t*,int); +}; + +void netcode_default_client_config( struct netcode_client_config_t * config ); + +struct netcode_client_t * netcode_client_create( NETCODE_CONST char * address, NETCODE_CONST struct netcode_client_config_t * config, double time ); + +void netcode_client_destroy( struct netcode_client_t * client ); + +void netcode_client_connect( struct netcode_client_t * client, uint8_t * connect_token ); + +void netcode_client_update( struct netcode_client_t * client, double time ); + +uint64_t netcode_client_next_packet_sequence( struct netcode_client_t * client ); + +void netcode_client_send_packet( struct netcode_client_t * client, NETCODE_CONST uint8_t * packet_data, int packet_bytes ); + +uint8_t * netcode_client_receive_packet( struct netcode_client_t * client, int * packet_bytes, uint64_t * packet_sequence ); + +void netcode_client_free_packet( struct netcode_client_t * client, void * packet ); + +void netcode_client_disconnect( struct netcode_client_t * client ); + +int netcode_client_state( struct netcode_client_t * client ); + +int netcode_client_index( struct netcode_client_t * client ); + +int netcode_client_max_clients( struct netcode_client_t * client ); + +void netcode_client_connect_loopback( struct netcode_client_t * client, int client_index, int max_clients ); + +void netcode_client_disconnect_loopback( struct netcode_client_t * client ); + +void netcode_client_process_packet( struct netcode_client_t * client, struct netcode_address_t * from, uint8_t * packet_data, int packet_bytes ); + +int netcode_client_loopback( struct netcode_client_t * client ); + +void netcode_client_process_loopback_packet( struct netcode_client_t * client, NETCODE_CONST uint8_t * packet_data, int packet_bytes, uint64_t packet_sequence ); + +uint16_t netcode_client_get_port( struct netcode_client_t * client ); + +struct netcode_address_t * netcode_client_server_address( struct netcode_client_t * client ); + +int netcode_generate_connect_token( int num_server_addresses, + NETCODE_CONST char ** public_server_addresses, + NETCODE_CONST char ** internal_server_addresses, + int expire_seconds, + int timeout_seconds, + uint64_t client_id, + uint64_t protocol_id, + NETCODE_CONST uint8_t * private_key, + uint8_t * user_data, + uint8_t * connect_token ); + +struct netcode_server_config_t +{ + uint64_t protocol_id; + uint8_t private_key[NETCODE_KEY_BYTES]; + void * allocator_context; + void * (*allocate_function)(void*,uint64_t); + void (*free_function)(void*,void*); + struct netcode_network_simulator_t * network_simulator; + void * callback_context; + void (*connect_disconnect_callback)(void*,int,int); + void (*send_loopback_packet_callback)(void*,int,NETCODE_CONST uint8_t*,int,uint64_t); + int override_send_and_receive; + void (*send_packet_override)(void*,struct netcode_address_t*,NETCODE_CONST uint8_t*,int); + int (*receive_packet_override)(void*,struct netcode_address_t*,uint8_t*,int); +}; + +void netcode_default_server_config( struct netcode_server_config_t * config ); + +struct netcode_server_t * netcode_server_create( NETCODE_CONST char * server_address, NETCODE_CONST struct netcode_server_config_t * config, double time ); + +void netcode_server_destroy( struct netcode_server_t * server ); + +void netcode_server_start( struct netcode_server_t * server, int max_clients ); + +void netcode_server_stop( struct netcode_server_t * server ); + +int netcode_server_running( struct netcode_server_t * server ); + +int netcode_server_max_clients( struct netcode_server_t * server ); + +void netcode_server_update( struct netcode_server_t * server, double time ); + +int netcode_server_client_connected( struct netcode_server_t * server, int client_index ); + +uint64_t netcode_server_client_id( struct netcode_server_t * server, int client_index ); + +struct netcode_address_t * netcode_server_client_address( struct netcode_server_t * server, int client_index ); + +void netcode_server_disconnect_client( struct netcode_server_t * server, int client_index ); + +void netcode_server_disconnect_all_clients( struct netcode_server_t * server ); + +uint64_t netcode_server_next_packet_sequence( struct netcode_server_t * server, int client_index ); + +void netcode_server_send_packet( struct netcode_server_t * server, int client_index, NETCODE_CONST uint8_t * packet_data, int packet_bytes ); + +uint8_t * netcode_server_receive_packet( struct netcode_server_t * server, int client_index, int * packet_bytes, uint64_t * packet_sequence ); + +void netcode_server_free_packet( struct netcode_server_t * server, void * packet ); + +int netcode_server_num_connected_clients( struct netcode_server_t * server ); + +void * netcode_server_client_user_data( struct netcode_server_t * server, int client_index ); + +void netcode_server_process_packet( struct netcode_server_t * server, struct netcode_address_t * from, uint8_t * packet_data, int packet_bytes ); + +void netcode_server_connect_loopback_client( struct netcode_server_t * server, int client_index, uint64_t client_id, NETCODE_CONST uint8_t * user_data ); + +void netcode_server_disconnect_loopback_client( struct netcode_server_t * server, int client_index ); + +int netcode_server_client_loopback( struct netcode_server_t * server, int client_index ); + +void netcode_server_process_loopback_packet( struct netcode_server_t * server, int client_index, NETCODE_CONST uint8_t * packet_data, int packet_bytes, uint64_t packet_sequence ); + +uint16_t netcode_server_get_port( struct netcode_server_t * server ); + +void netcode_log_level( int level ); + +void netcode_set_printf_function( int (*function)( NETCODE_CONST char *, ... ) ); + +extern void (*netcode_assert_function)( NETCODE_CONST char *, NETCODE_CONST char *, NETCODE_CONST char * file, int line ); + +#ifndef NDEBUG +#define netcode_assert( condition ) \ +do \ +{ \ + if ( !(condition) ) \ + { \ + netcode_assert_function( #condition, __FUNCTION__, __FILE__, __LINE__ ); \ + exit(1); \ + } \ +} while(0) +#else +#define netcode_assert( ignore ) ((void)0) +#endif + +void netcode_set_assert_function( void (*function)( NETCODE_CONST char * /*condition*/, + NETCODE_CONST char * /*function*/, + NETCODE_CONST char * /*file*/, + int /*line*/ ) ); + +void netcode_random_bytes( uint8_t * data, int bytes ); + +void netcode_sleep( double seconds ); + +double netcode_time(); + +#ifdef __cplusplus +} +#endif + +#endif // #ifndef NETCODE_H diff --git a/premake5.lua b/premake5.lua index c4fe7530..75f638b4 100644 --- a/premake5.lua +++ b/premake5.lua @@ -1,432 +1,50 @@ yojimbo_version = "1.0" -libs = { "sodium", "mbedtls", "mbedx509", "mbedcrypto" } - solution "Yojimbo" kind "ConsoleApp" language "C++" - platforms { "x64" } configurations { "Debug", "Release" } + includedirs { ".", "mbedtls", "sodium", "tlsf", "certs", "netcode", "reliable" } if os.istarget "windows" then - includedirs { ".", "./windows", "netcode.io", "reliable.io" } libdirs { "./windows" } else - includedirs { ".", "/usr/local/include", "netcode.io", "reliable.io" } targetdir "bin/" end rtti "Off" - links { libs } warnings "Extra" floatingpoint "Fast" - vectorextensions "SSE2" configuration "Debug" symbols "On" defines { "YOJIMBO_DEBUG", "NETCODE_DEBUG", "RELIABLE_DEBUG" } configuration "Release" optimize "Speed" defines { "YOJIMBO_RELEASE", "NETCODE_RELEASE", "RELIABLE_RELEASE" } - -project "test" - files { "test.cpp" } - links { "yojimbo" } -project "yojimbo" +project "netcode" kind "StaticLib" - defines { "NETCODE_ENABLE_TESTS=1", "RELIABLE_ENABLE_TESTS=1" } - files { "yojimbo.h", "yojimbo.cpp", "certs.h", "certs.c", "tlsf/tlsf.h", "tlsf/tlsf.c", "netcode.io/netcode.c", "netcode.io/netcode.h", "reliable.io/reliable.c", "reliable.io/reliable.h" } - -project "client" - files { "client.cpp", "shared.h" } - links { "yojimbo" } - -project "server" - files { "server.cpp", "shared.h" } - links { "yojimbo" } - -project "secure_client" - files { "secure_client.cpp", "shared.h" } - links { "yojimbo" } - -project "secure_server" - files { "secure_server.cpp", "shared.h" } - links { "yojimbo" } - -project "client_server" - files { "client_server.cpp", "shared.h" } - links { "yojimbo" } - -project "loopback" - files { "loopback.cpp", "shared.h" } - links { "yojimbo" } - -project "soak" - files { "soak.cpp", "shared.h" } - links { "yojimbo" } - -if not os.istarget "windows" then - - -- MacOSX and Linux. - - newaction - { - trigger = "test", - description = "Build and run all unit tests", - execute = function () - os.execute "test ! -e Makefile && premake5 gmake2" - if os.execute "make -j32 test" then - os.execute "./bin/test" - end - end - } - - newaction - { - trigger = "client_server", - description = "Build and run client/server test", - execute = function () - os.execute "test ! -e Makefile && premake5 gmake2" - if os.execute "make -j32 client_server" then - os.execute "./bin/client_server" - end - end - } - - newaction - { - trigger = "loopback", - description = "Build and run loopback test", - execute = function () - os.execute "test ! -e Makefile && premake5 gmake2" - if os.execute "make -j32 loopback" then - os.execute "./bin/loopback" - end - end - } - - newoption - { - trigger = "serverAddress", - value = "IP[:port]", - description = "Specify the server address that the client should connect to", - } - - newaction - { - trigger = "client", - description = "Build and run client", - valid_kinds = premake.action.get("gmake2").valid_kinds, - valid_languages = premake.action.get("gmake2").valid_languages, - valid_tools = premake.action.get("gmake2").valid_tools, - - execute = function () - os.execute "test ! -e Makefile && premake5 gmake2" - if os.execute "make -j32 client" then - if _OPTIONS["serverAddress"] then - os.execute( "./bin/client " .. _OPTIONS["serverAddress"] ) - else - os.execute "./bin/client" - end - end - end - } - - newaction - { - trigger = "server", - description = "Build and run server", - execute = function () - os.execute "test ! -e Makefile && premake5 gmake2" - if os.execute "make -j32 server" then - os.execute "./bin/server" - end - end - } - - newaction - { - trigger = "secure_server", - description = "Build and run secure server", - execute = function () - os.execute "test ! -e Makefile && premake5 gmake2" - if os.execute "make -j32 secure_server" then - os.execute "./bin/secure_server" - end - end - } - - newaction - { - trigger = "docker", - description = "Build and run a yojimbo server inside a docker container", - execute = function () - os.execute "docker run --rm --privileged alpine hwclock -s" -- workaround for clock getting out of sync on macos. see https://docs.docker.com/docker-for-mac/troubleshoot/#issues - os.execute "rm -rf docker/yojimbo && mkdir -p docker/yojimbo && mkdir -p docker/yojimbo/tests && cp *.h docker/yojimbo && cp *.cpp docker/yojimbo && cp premake5.lua docker/yojimbo && cp -R reliable.io docker/yojimbo && cp -R netcode.io docker/yojimbo && cp -R tlsf docker/yojimbo && cd docker && docker build -t \"networkprotocol:yojimbo-server\" . && rm -rf yojimbo && docker run -ti -p 40000:40000/udp networkprotocol:yojimbo-server" - end - } - - newaction - { - trigger = "valgrind", - description = "Run valgrind over tests inside docker", - execute = function () - os.execute "rm -rf valgrind/yojimbo && mkdir -p valgrind/yojimbo && mkdir -p valgrind/yojimbo/tests && cp *.h valgrind/yojimbo && cp *.cpp valgrind/yojimbo && cp premake5.lua valgrind/yojimbo && cp -R reliable.io valgrind/yojimbo && cp -R netcode.io valgrind/yojimbo && cp -R tlsf valgrind/yojimbo && cd valgrind && docker build -t \"networkprotocol:yojimbo-valgrind\" . && rm -rf netcode.io && docker run -ti networkprotocol:yojimbo-valgrind" - end - } - - newaction - { - trigger = "matcher", - description = "Build and run the matchmaker web service inside a docker container", - execute = function () - os.execute "docker run --rm --privileged alpine hwclock -s" -- workaround for clock getting out of sync on macos. see https://docs.docker.com/docker-for-mac/troubleshoot/#issues - os.execute "cd matcher && docker build -t networkprotocol:yojimbo-matcher . && docker run -ti -p 8080:8080 networkprotocol:yojimbo-matcher" - end - } - - newaction - { - trigger = "secure_client", - description = "Build and run secure client and connect to a server via the matcher", - execute = function () - os.execute "test ! -e Makefile && premake5 gmake2" - if os.execute "make -j32 secure_client" then - os.execute "./bin/secure_client" - end - end - } - - newaction - { - trigger = "stress", - description = "Launch 64 secure client instances to stress the matcher and server", - execute = function () - os.execute "test ! -e Makefile && premake5 gmake2" - if os.execute "make -j32 secure_client" then - for i = 0, 63 do - os.execute "./bin/secure_client &" - end - end - end - } - - newaction - { - trigger = "soak", - description = "Build and run soak test", - execute = function () - os.execute "test ! -e Makefile && premake5 gmake2" - if os.execute "make -j32 soak" then - os.execute "./bin/soak" - end - end - } - - newaction - { - trigger = "cppcheck", - description = "Run cppcheck over the source code and write to cppcheck.txt", - execute = function () - os.execute "cppcheck *.h *.cpp --force --std=c++03 --language=c++ --quiet -U min -U max 2>&1 --config-exclude=tlsf --suppress=incorrectStringBooleanError --suppress=cstyleCast --suppress=unusedFunction --suppress=unusedStructMember --suppress=variableScope --suppress=memsetClassFloat --enable=warning --enable=performance --enable=style --platform=native -j 32 | tee -a cppcheck.txt" - end - } - - newaction - { - trigger = "scan-build", - description = "Run clang scan-build over the project", - execute = function () - os.execute "premake5 clean && premake5 gmake2 && scan-build make all -j32" - end - } + language "C" + defines { "NETCODE_ENABLE_TESTS=1" } + files { "netcode/netcode.c", "netcode/netcode.h" } - newaction - { - trigger = "coverity", - description = "Integrate latest code into coverity_scan so it gets coverity scanned by travis job", - execute = function () - os.execute "git checkout coverity_scan && git merge master && git push && git checkout master" - end - } - - newaction - { - trigger = "loc", - description = "Count lines of code", - execute = function () - os.execute "wc -l *.h *.cpp netcode.io/*.c netcode.io/*.h reliable.io/*.c reliable.io/*.h" - end - } - - newaction - { - trigger = "release", - description = "Create a release of this project", - execute = function () - _ACTION = "clean" - premake.action.call( "clean" ) - files_to_zip = "README.md BUILDING.md CHANGES.md ROADMAP.md *.cpp *.h premake5.lua docker tests tlsf windows" - -- todo: need to update this so it works with netcode.io and reliable.io sub-projects - os.execute( "rm -rf *.zip *.tar.gz" ); - os.execute( "rm -rf docker/yojimbo" ); - os.execute( "zip -9r yojimbo-" .. yojimbo_version .. ".zip " .. files_to_zip ) - os.execute( "unzip yojimbo-" .. yojimbo_version .. ".zip -d yojimbo-" .. yojimbo_version ); - os.execute( "tar -zcvf yojimbo-" .. yojimbo_version .. ".tar.gz yojimbo-" .. yojimbo_version ); - os.execute( "rm -rf yojimbo-" .. yojimbo_version ); - os.execute( "mkdir -p release" ); - os.execute( "mv yojimbo-" .. yojimbo_version .. ".zip release" ); - os.execute( "mv yojimbo-" .. yojimbo_version .. ".tar.gz release" ); - os.execute( "echo" ); - os.execute( "echo \"*** SUCCESSFULLY CREATED RELEASE - yojimbo-" .. yojimbo_version .. " *** \"" ); - os.execute( "echo" ); - end - } - - newaction - { - trigger = "sublime", - description = "Create sublime project", - execute = function () - os.execute "cp .sublime yojimbo.sublime-project" - end - } - - newaction - { - trigger = "docs", - description = "Build documentation", - execute = function () - if os.host() == "macosx" then - os.execute "doxygen doxygen.config && open docs/html/index.html" - else - os.execute "doxygen doxygen.config" - end - end - } - - newaction - { - trigger = "update_submodules", - description = "Updates to latest code for netcode.io and reliable.io", - execute = function () - os.execute "git pull" - os.execute "git submodule update --remote --merge" - os.execute "git add *" - os.execute "git commit -am \"update submodules\"" - os.execute "git push" - end - } - -else - - -- Windows - - newaction - { - trigger = "solution", - description = "Open Yojimbo.sln", - execute = function () - os.execute "premake5 vs2019" - os.execute "start Yojimbo.sln" - end - } - - newaction - { - trigger = "docker", - description = "Build and run a yojimbo server inside a docker container", - execute = function () - os.execute "cd docker && copyFiles.bat && buildServer.bat && runServer.bat" - end - } - - newaction - { - trigger = "matcher", - description = "Build and run the matchmaker web service inside a docker container", - execute = function () - os.execute "cd matcher && docker build -t networkprotocol:yojimbo-matcher . && docker run -ti -p 8080:8080 networkprotocol:yojimbo-matcher" - end - } - - newaction - { - trigger = "stress", - description = "Launch 64 secure client instances to stress the matcher and server", - execute = function () - for i = 0, 63 do - os.execute "if exist bin\\x64\\Debug\\secure_client.exe ( start /B bin\\x64\\Debug\\secure_client ) else ( echo could not find bin\\x64\\Debug\\secure_client.exe )" - end - end - } - - newaction - { - trigger = "docs", - description = "Build documentation", - execute = function () - os.execute "doxygen doxygen.config && start docs\\html\\index.html" - end - } - -end - -newaction -{ - trigger = "clean", - - description = "Clean all build files and output", - - execute = function () - - files_to_delete = - { - "Makefile", - "*.make", - "*.txt", - "*.zip", - "*.tar.gz", - "*.db", - "*.opendb", - "*.vcproj", - "*.vcxproj", - "*.vcxproj.user", - "*.vcxproj.filters", - "*.sln", - "*.xcodeproj", - "*.xcworkspace" - } - - directories_to_delete = - { - "obj", - "ipch", - "bin", - ".vs", - "Debug", - "Release", - "release", - "cov-int", - "docker/yojimbo", - "valgrind/yojimbo", - "docs", - "xml" - } +project "reliable" + kind "StaticLib" + language "C" + defines { "NETCODE_ENABLE_TESTS=1" } + files { "reliable/reliable.c", "reliable/reliable.h" } - for i,v in ipairs( directories_to_delete ) do - os.rmdir( v ) - end +project "tslf" + kind "StaticLib" + language "C" + files { "tlsf/tlsf.c", "tlsf/tlsf.h" } - if not os.istarget "windows" then - os.execute "find . -name .DS_Store -delete" - for i,v in ipairs( files_to_delete ) do - os.execute( "rm -f " .. v ) - end - else - for i,v in ipairs( files_to_delete ) do - os.execute( "del /F /Q " .. v ) - end - end +project "certs" + kind "StaticLib" + language "C" + files { "certs/certs.c", "certs/certs.h" } - end -} +project "yojimbo" + kind "StaticLib" + links { "reliable", "netcode", "tlsf", "certs" } + defines { "NETCODE_ENABLE_TESTS=1", "RELIABLE_ENABLE_TESTS=1" } + files { "yojimbo.h", "yojimbo.cpp" } diff --git a/reliable.io b/reliable.io deleted file mode 160000 index f5d12957..00000000 --- a/reliable.io +++ /dev/null @@ -1 +0,0 @@ -Subproject commit f5d12957c3b53ed5ba15b7dd873c95cf2c6e4de9 diff --git a/reliable/reliable.c b/reliable/reliable.c new file mode 100644 index 00000000..818af2fd --- /dev/null +++ b/reliable/reliable.c @@ -0,0 +1,2418 @@ +/* + reliable + + Copyright © 2017 - 2024, Mas Bandwidth LLC + + Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: + + 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the distribution. + + 3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived + from this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, + INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, + WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE + USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ + +#include "reliable.h" +#include +#include +#include +#include +#include +#include +#include +#include + +#ifndef RELIABLE_ENABLE_TESTS +#define RELIABLE_ENABLE_TESTS 0 +#endif // #ifndef RELIABLE_ENABLE_TESTS + +#ifndef RELIABLE_ENABLE_LOGGING +#define RELIABLE_ENABLE_LOGGING 1 +#endif // #ifndef RELIABLE_ENABLE_LOGGING + +// ------------------------------------------------------------------ + +static void default_assert_handler( RELIABLE_CONST char * condition, RELIABLE_CONST char * function, RELIABLE_CONST char * file, int line ) +{ + printf( "assert failed: ( %s ), function %s, file %s, line %d\n", condition, function, file, line ); + #if defined( __GNUC__ ) + __builtin_trap(); + #elif defined( _MSC_VER ) + __debugbreak(); + #endif + exit( 1 ); +} + +static int log_level = 0; +static int (*printf_function)( RELIABLE_CONST char *, ... ) = ( int (*)( RELIABLE_CONST char *, ... ) ) printf; +void (*reliable_assert_function)( RELIABLE_CONST char *, RELIABLE_CONST char *, RELIABLE_CONST char * file, int line ) = default_assert_handler; + +void reliable_log_level( int level ) +{ + log_level = level; +} + +void reliable_set_printf_function( int (*function)( RELIABLE_CONST char *, ... ) ) +{ + reliable_assert( function ); + printf_function = function; +} + +void reliable_set_assert_function( void (*function)( RELIABLE_CONST char *, RELIABLE_CONST char *, RELIABLE_CONST char * file, int line ) ) +{ + reliable_assert_function = function; +} + +#if RELIABLE_ENABLE_LOGGING + +void reliable_printf( int level, RELIABLE_CONST char * format, ... ) +{ + if ( level > log_level ) + return; + va_list args; + va_start( args, format ); + char buffer[4*1024]; + vsnprintf( buffer, sizeof(buffer), format, args ); + printf_function( "%s", buffer ); + va_end( args ); +} + +#else // #if RELIABLE_ENABLE_LOGGING + +void reliable_printf( int level, RELIABLE_CONST char * format, ... ) +{ + (void) level; + (void) format; +} + +#endif // #if RELIABLE_ENABLE_LOGGING + +void * reliable_default_allocate_function( void * context, uint64_t bytes ) +{ + (void) context; + return malloc( bytes ); +} + +void reliable_default_free_function( void * context, void * pointer ) +{ + (void) context; + free( pointer ); +} + +// ------------------------------------------------------------------ + +int reliable_init(void) +{ + return RELIABLE_OK; +} + +void reliable_term(void) +{ +} + +// --------------------------------------------------------------- + +int reliable_sequence_greater_than( uint16_t s1, uint16_t s2 ) +{ + return ( ( s1 > s2 ) && ( s1 - s2 <= 32768 ) ) || + ( ( s1 < s2 ) && ( s2 - s1 > 32768 ) ); +} + +int reliable_sequence_less_than( uint16_t s1, uint16_t s2 ) +{ + return reliable_sequence_greater_than( s2, s1 ); +} + +// --------------------------------------------------------------- + +struct reliable_sequence_buffer_t +{ + void * allocator_context; + void * (*allocate_function)(void*,uint64_t); + void (*free_function)(void*,void*); + uint16_t sequence; + int num_entries; + int entry_stride; + uint32_t * entry_sequence; + uint8_t * entry_data; +}; + +struct reliable_sequence_buffer_t * reliable_sequence_buffer_create( int num_entries, + int entry_stride, + void * allocator_context, + void * (*allocate_function)(void*,uint64_t), + void (*free_function)(void*,void*) ) +{ + reliable_assert( num_entries > 0 ); + reliable_assert( entry_stride > 0 ); + + if ( allocate_function == NULL ) + { + allocate_function = reliable_default_allocate_function; + } + + if ( free_function == NULL ) + { + free_function = reliable_default_free_function; + } + + struct reliable_sequence_buffer_t * sequence_buffer = (struct reliable_sequence_buffer_t*) + allocate_function( allocator_context, sizeof( struct reliable_sequence_buffer_t ) ); + + sequence_buffer->allocator_context = allocator_context; + sequence_buffer->allocate_function = allocate_function; + sequence_buffer->free_function = free_function; + sequence_buffer->sequence = 0; + sequence_buffer->num_entries = num_entries; + sequence_buffer->entry_stride = entry_stride; + sequence_buffer->entry_sequence = (uint32_t*) allocate_function( allocator_context, num_entries * sizeof( uint32_t ) ); + sequence_buffer->entry_data = (uint8_t*) allocate_function( allocator_context, num_entries * entry_stride ); + reliable_assert( sequence_buffer->entry_sequence ); + reliable_assert( sequence_buffer->entry_data ); + memset( sequence_buffer->entry_sequence, 0xFF, sizeof( uint32_t) * sequence_buffer->num_entries ); + memset( sequence_buffer->entry_data, 0, num_entries * entry_stride ); + + return sequence_buffer; +} + +void reliable_sequence_buffer_destroy( struct reliable_sequence_buffer_t * sequence_buffer ) +{ + reliable_assert( sequence_buffer ); + sequence_buffer->free_function( sequence_buffer->allocator_context, sequence_buffer->entry_sequence ); + sequence_buffer->free_function( sequence_buffer->allocator_context, sequence_buffer->entry_data ); + sequence_buffer->free_function( sequence_buffer->allocator_context, sequence_buffer ); +} + +void reliable_sequence_buffer_reset( struct reliable_sequence_buffer_t * sequence_buffer ) +{ + reliable_assert( sequence_buffer ); + sequence_buffer->sequence = 0; + memset( sequence_buffer->entry_sequence, 0xFF, sizeof( uint32_t) * sequence_buffer->num_entries ); +} + +void reliable_sequence_buffer_remove_entries( struct reliable_sequence_buffer_t * sequence_buffer, + int start_sequence, + int finish_sequence, + void (*cleanup_function)(void*,void*,void(*free_function)(void*,void*)) ) +{ + reliable_assert( sequence_buffer ); + if ( finish_sequence < start_sequence ) + { + finish_sequence += 65536; + } + if ( finish_sequence - start_sequence < sequence_buffer->num_entries ) + { + int sequence; + for ( sequence = start_sequence; sequence <= finish_sequence; ++sequence ) + { + if ( cleanup_function ) + { + cleanup_function( sequence_buffer->entry_data + sequence_buffer->entry_stride * ( sequence % sequence_buffer->num_entries ), + sequence_buffer->allocator_context, + sequence_buffer->free_function ); + } + sequence_buffer->entry_sequence[ sequence % sequence_buffer->num_entries ] = 0xFFFFFFFF; + } + } + else + { + int i; + for ( i = 0; i < sequence_buffer->num_entries; ++i ) + { + if ( cleanup_function ) + { + cleanup_function( sequence_buffer->entry_data + sequence_buffer->entry_stride * i, + sequence_buffer->allocator_context, + sequence_buffer->free_function ); + } + sequence_buffer->entry_sequence[i] = 0xFFFFFFFF; + } + } +} + +int reliable_sequence_buffer_test_insert( struct reliable_sequence_buffer_t * sequence_buffer, uint16_t sequence ) +{ + return reliable_sequence_less_than( sequence, sequence_buffer->sequence - ((uint16_t)sequence_buffer->num_entries) ) ? ((uint16_t)0) : ((uint16_t)1); +} + +void * reliable_sequence_buffer_insert( struct reliable_sequence_buffer_t * sequence_buffer, uint16_t sequence ) +{ + reliable_assert( sequence_buffer ); + if ( reliable_sequence_less_than( sequence, sequence_buffer->sequence - ((uint16_t)sequence_buffer->num_entries) ) ) + { + return NULL; + } + if ( reliable_sequence_greater_than( sequence + 1, sequence_buffer->sequence ) ) + { + reliable_sequence_buffer_remove_entries( sequence_buffer, sequence_buffer->sequence, sequence, NULL ); + sequence_buffer->sequence = sequence + 1; + } + int index = sequence % sequence_buffer->num_entries; + sequence_buffer->entry_sequence[index] = sequence; + return sequence_buffer->entry_data + index * sequence_buffer->entry_stride; +} + +void reliable_sequence_buffer_advance( struct reliable_sequence_buffer_t * sequence_buffer, uint16_t sequence ) +{ + reliable_assert( sequence_buffer ); + if ( reliable_sequence_greater_than( sequence + 1, sequence_buffer->sequence ) ) + { + reliable_sequence_buffer_remove_entries( sequence_buffer, sequence_buffer->sequence, sequence, NULL ); + sequence_buffer->sequence = sequence + 1; + } +} + +void * reliable_sequence_buffer_insert_with_cleanup( struct reliable_sequence_buffer_t * sequence_buffer, + uint16_t sequence, + void (*cleanup_function)(void*,void*,void(*free_function)(void*,void*)) ) +{ + reliable_assert( sequence_buffer ); + if ( reliable_sequence_greater_than( sequence + 1, sequence_buffer->sequence ) ) + { + reliable_sequence_buffer_remove_entries( sequence_buffer, sequence_buffer->sequence, sequence, cleanup_function ); + sequence_buffer->sequence = sequence + 1; + } + else if ( reliable_sequence_less_than( sequence, sequence_buffer->sequence - ((uint16_t)sequence_buffer->num_entries) ) ) + { + return NULL; + } + int index = sequence % sequence_buffer->num_entries; + if ( sequence_buffer->entry_sequence[index] != 0xFFFFFFFF ) + { + cleanup_function( sequence_buffer->entry_data + sequence_buffer->entry_stride * ( sequence % sequence_buffer->num_entries ), + sequence_buffer->allocator_context, + sequence_buffer->free_function ); + } + sequence_buffer->entry_sequence[index] = sequence; + return sequence_buffer->entry_data + index * sequence_buffer->entry_stride; +} + +void reliable_sequence_buffer_advance_with_cleanup( struct reliable_sequence_buffer_t * sequence_buffer, + uint16_t sequence, + void (*cleanup_function)(void*,void*,void(*free_function)(void*,void*)) ) +{ + reliable_assert( sequence_buffer ); + if ( reliable_sequence_greater_than( sequence + 1, sequence_buffer->sequence ) ) + { + reliable_sequence_buffer_remove_entries( sequence_buffer, sequence_buffer->sequence, sequence, cleanup_function ); + sequence_buffer->sequence = sequence + 1; + } +} + +void reliable_sequence_buffer_remove( struct reliable_sequence_buffer_t * sequence_buffer, uint16_t sequence ) +{ + reliable_assert( sequence_buffer ); + sequence_buffer->entry_sequence[ sequence % sequence_buffer->num_entries ] = 0xFFFFFFFF; +} + +void reliable_sequence_buffer_remove_with_cleanup( struct reliable_sequence_buffer_t * sequence_buffer, + uint16_t sequence, + void (*cleanup_function)(void*,void*,void(*free_function)(void*,void*)) ) +{ + reliable_assert( sequence_buffer ); + int index = sequence % sequence_buffer->num_entries; + if ( sequence_buffer->entry_sequence[index] != 0xFFFFFFFF ) + { + sequence_buffer->entry_sequence[index] = 0xFFFFFFFF; + cleanup_function( sequence_buffer->entry_data + sequence_buffer->entry_stride * index, sequence_buffer->allocator_context, sequence_buffer->free_function ); + } +} + +int reliable_sequence_buffer_available( struct reliable_sequence_buffer_t * sequence_buffer, uint16_t sequence ) +{ + reliable_assert( sequence_buffer ); + return sequence_buffer->entry_sequence[ sequence % sequence_buffer->num_entries ] == 0xFFFFFFFF; +} + +int reliable_sequence_buffer_exists( struct reliable_sequence_buffer_t * sequence_buffer, uint16_t sequence ) +{ + reliable_assert( sequence_buffer ); + return sequence_buffer->entry_sequence[ sequence % sequence_buffer->num_entries ] == (uint32_t) sequence; +} + +void * reliable_sequence_buffer_find( struct reliable_sequence_buffer_t * sequence_buffer, uint16_t sequence ) +{ + reliable_assert( sequence_buffer ); + int index = sequence % sequence_buffer->num_entries; + return ( ( sequence_buffer->entry_sequence[index] == (uint32_t) sequence ) ) ? ( sequence_buffer->entry_data + index * sequence_buffer->entry_stride ) : NULL; + +} + +void * reliable_sequence_buffer_at_index( struct reliable_sequence_buffer_t * sequence_buffer, int index ) +{ + reliable_assert( sequence_buffer ); + reliable_assert( index >= 0 ); + reliable_assert( index < sequence_buffer->num_entries ); + return sequence_buffer->entry_sequence[index] != 0xFFFFFFFF ? ( sequence_buffer->entry_data + index * sequence_buffer->entry_stride ) : NULL; +} + +void reliable_sequence_buffer_generate_ack_bits( struct reliable_sequence_buffer_t * sequence_buffer, uint16_t * ack, uint32_t * ack_bits ) +{ + reliable_assert( sequence_buffer ); + reliable_assert( ack ); + reliable_assert( ack_bits ); + *ack = sequence_buffer->sequence - 1; + *ack_bits = 0; + uint32_t mask = 1; + int i; + for ( i = 0; i < 32; ++i ) + { + uint16_t sequence = *ack - ((uint16_t)i); + if ( reliable_sequence_buffer_exists( sequence_buffer, sequence ) ) + *ack_bits |= mask; + mask <<= 1; + } +} + +// --------------------------------------------------------------- + +void reliable_write_uint8( uint8_t ** p, uint8_t value ) +{ + **p = value; + ++(*p); +} + +void reliable_write_uint16( uint8_t ** p, uint16_t value ) +{ + (*p)[0] = value & 0xFF; + (*p)[1] = value >> 8; + *p += 2; +} + +void reliable_write_uint32( uint8_t ** p, uint32_t value ) +{ + (*p)[0] = value & 0xFF; + (*p)[1] = ( value >> 8 ) & 0xFF; + (*p)[2] = ( value >> 16 ) & 0xFF; + (*p)[3] = value >> 24; + *p += 4; +} + +void reliable_write_uint64( uint8_t ** p, uint64_t value ) +{ + (*p)[0] = value & 0xFF; + (*p)[1] = ( value >> 8 ) & 0xFF; + (*p)[2] = ( value >> 16 ) & 0xFF; + (*p)[3] = ( value >> 24 ) & 0xFF; + (*p)[4] = ( value >> 32 ) & 0xFF; + (*p)[5] = ( value >> 40 ) & 0xFF; + (*p)[6] = ( value >> 48 ) & 0xFF; + (*p)[7] = value >> 56; + *p += 8; +} + +void reliable_write_bytes( uint8_t ** p, uint8_t * byte_array, int num_bytes ) +{ + int i; + for ( i = 0; i < num_bytes; ++i ) + { + reliable_write_uint8( p, byte_array[i] ); + } +} + +uint8_t reliable_read_uint8( uint8_t ** p ) +{ + uint8_t value = **p; + ++(*p); + return value; +} + +uint16_t reliable_read_uint16( uint8_t ** p ) +{ + uint16_t value; + value = (*p)[0]; + value |= ( ( (uint16_t)( (*p)[1] ) ) << 8 ); + *p += 2; + return value; +} + +uint32_t reliable_read_uint32( uint8_t ** p ) +{ + uint32_t value; + value = (*p)[0]; + value |= ( ( (uint32_t)( (*p)[1] ) ) << 8 ); + value |= ( ( (uint32_t)( (*p)[2] ) ) << 16 ); + value |= ( ( (uint32_t)( (*p)[3] ) ) << 24 ); + *p += 4; + return value; +} + +uint64_t reliable_read_uint64( uint8_t ** p ) +{ + uint64_t value; + value = (*p)[0]; + value |= ( ( (uint64_t)( (*p)[1] ) ) << 8 ); + value |= ( ( (uint64_t)( (*p)[2] ) ) << 16 ); + value |= ( ( (uint64_t)( (*p)[3] ) ) << 24 ); + value |= ( ( (uint64_t)( (*p)[4] ) ) << 32 ); + value |= ( ( (uint64_t)( (*p)[5] ) ) << 40 ); + value |= ( ( (uint64_t)( (*p)[6] ) ) << 48 ); + value |= ( ( (uint64_t)( (*p)[7] ) ) << 56 ); + *p += 8; + return value; +} + +void reliable_read_bytes( uint8_t ** p, uint8_t * byte_array, int num_bytes ) +{ + int i; + for ( i = 0; i < num_bytes; ++i ) + { + byte_array[i] = reliable_read_uint8( p ); + } +} + +// --------------------------------------------------------------- + +struct reliable_fragment_reassembly_data_t +{ + uint16_t sequence; + uint16_t ack; + uint32_t ack_bits; + int num_fragments_received; + int num_fragments_total; + uint8_t * packet_data; + int packet_bytes; + int packet_header_bytes; + uint8_t fragment_received[256]; +}; + +void reliable_fragment_reassembly_data_cleanup( void * data, void * allocator_context, void (*free_function)(void*,void*) ) + +{ + reliable_assert( free_function ); + struct reliable_fragment_reassembly_data_t * reassembly_data = (struct reliable_fragment_reassembly_data_t*) data; + if ( reassembly_data->packet_data ) + { + free_function( allocator_context, reassembly_data->packet_data ); + reassembly_data->packet_data = NULL; + } +} + +// --------------------------------------------------------------- + +struct reliable_endpoint_t +{ + void * allocator_context; + void * (*allocate_function)(void*,uint64_t); + void (*free_function)(void*,void*); + struct reliable_config_t config; + double time; + float rtt; + float packet_loss; + float sent_bandwidth_kbps; + float received_bandwidth_kbps; + float acked_bandwidth_kbps; + int num_acks; + uint16_t * acks; + uint16_t sequence; + struct reliable_sequence_buffer_t * sent_packets; + struct reliable_sequence_buffer_t * received_packets; + struct reliable_sequence_buffer_t * fragment_reassembly; + uint64_t counters[RELIABLE_ENDPOINT_NUM_COUNTERS]; +}; + +struct reliable_sent_packet_data_t +{ + double time; + uint32_t acked : 1; + uint32_t packet_bytes : 31; +}; + +struct reliable_received_packet_data_t +{ + double time; + uint32_t packet_bytes; +}; + +void reliable_default_config( struct reliable_config_t * config ) +{ + reliable_assert( config ); + memset( config, 0, sizeof( struct reliable_config_t ) ); + config->name[0] = 'e'; + config->name[1] = 'n'; + config->name[2] = 'd'; + config->name[3] = 'p'; + config->name[4] = 'o'; + config->name[5] = 'i'; + config->name[6] = 'n'; + config->name[7] = 't'; + config->name[8] = '\0'; + config->max_packet_size = 16 * 1024; + config->fragment_above = 1024; + config->max_fragments = 16; + config->fragment_size = 1024; + config->ack_buffer_size = 256; + config->sent_packets_buffer_size = 256; + config->received_packets_buffer_size = 256; + config->fragment_reassembly_buffer_size = 64; + config->rtt_smoothing_factor = 0.0025f; + config->packet_loss_smoothing_factor = 0.1f; + config->bandwidth_smoothing_factor = 0.1f; + config->packet_header_size = 28; // note: UDP over IPv4 = 20 + 8 bytes, UDP over IPv6 = 40 + 8 bytes +} + +struct reliable_endpoint_t * reliable_endpoint_create( struct reliable_config_t * config, double time ) +{ + reliable_assert( config ); + reliable_assert( config->max_packet_size > 0 ); + reliable_assert( config->fragment_above > 0 ); + reliable_assert( config->max_fragments > 0 ); + reliable_assert( config->max_fragments <= 256 ); + reliable_assert( config->fragment_size > 0 ); + reliable_assert( config->ack_buffer_size > 0 ); + reliable_assert( config->sent_packets_buffer_size > 0 ); + reliable_assert( config->received_packets_buffer_size > 0 ); + reliable_assert( config->transmit_packet_function != NULL ); + reliable_assert( config->process_packet_function != NULL ); + + void * allocator_context = config->allocator_context; + void * (*allocate_function)(void*,uint64_t) = config->allocate_function; + void (*free_function)(void*,void*) = config->free_function; + + if ( allocate_function == NULL ) + { + allocate_function = reliable_default_allocate_function; + } + + if ( free_function == NULL ) + { + free_function = reliable_default_free_function; + } + + struct reliable_endpoint_t * endpoint = (struct reliable_endpoint_t*) allocate_function( allocator_context, sizeof( struct reliable_endpoint_t ) ); + + reliable_assert( endpoint ); + + memset( endpoint, 0, sizeof( struct reliable_endpoint_t ) ); + + endpoint->allocator_context = allocator_context; + endpoint->allocate_function = allocate_function; + endpoint->free_function = free_function; + endpoint->config = *config; + endpoint->time = time; + + endpoint->acks = (uint16_t*) allocate_function( allocator_context, config->ack_buffer_size * sizeof( uint16_t ) ); + + endpoint->sent_packets = reliable_sequence_buffer_create( config->sent_packets_buffer_size, + sizeof( struct reliable_sent_packet_data_t ), + allocator_context, + allocate_function, + free_function ); + + endpoint->received_packets = reliable_sequence_buffer_create( config->received_packets_buffer_size, + sizeof( struct reliable_received_packet_data_t ), + allocator_context, + allocate_function, + free_function ); + + endpoint->fragment_reassembly = reliable_sequence_buffer_create( config->fragment_reassembly_buffer_size, + sizeof( struct reliable_fragment_reassembly_data_t ), + allocator_context, + allocate_function, + free_function ); + + memset( endpoint->acks, 0, config->ack_buffer_size * sizeof( uint16_t ) ); + + return endpoint; +} + +void reliable_endpoint_destroy( struct reliable_endpoint_t * endpoint ) +{ + reliable_assert( endpoint ); + reliable_assert( endpoint->acks ); + reliable_assert( endpoint->sent_packets ); + reliable_assert( endpoint->received_packets ); + + int i; + for ( i = 0; i < endpoint->config.fragment_reassembly_buffer_size; ++i ) + { + struct reliable_fragment_reassembly_data_t * reassembly_data = (struct reliable_fragment_reassembly_data_t*) + reliable_sequence_buffer_at_index( endpoint->fragment_reassembly, i ); + + if ( reassembly_data && reassembly_data->packet_data ) + { + endpoint->free_function( endpoint->allocator_context, reassembly_data->packet_data ); + reassembly_data->packet_data = NULL; + } + } + + endpoint->free_function( endpoint->allocator_context, endpoint->acks ); + + reliable_sequence_buffer_destroy( endpoint->sent_packets ); + reliable_sequence_buffer_destroy( endpoint->received_packets ); + reliable_sequence_buffer_destroy( endpoint->fragment_reassembly ); + + endpoint->free_function( endpoint->allocator_context, endpoint ); +} + +uint16_t reliable_endpoint_next_packet_sequence( struct reliable_endpoint_t * endpoint ) +{ + reliable_assert( endpoint ); + return endpoint->sequence; +} + +int reliable_write_packet_header( uint8_t * packet_data, uint16_t sequence, uint16_t ack, uint32_t ack_bits ) +{ + uint8_t * p = packet_data; + + uint8_t prefix_byte = 0; + + if ( ( ack_bits & 0x000000FF ) != 0x000000FF ) + { + prefix_byte |= (1<<1); + } + + if ( ( ack_bits & 0x0000FF00 ) != 0x0000FF00 ) + { + prefix_byte |= (1<<2); + } + + if ( ( ack_bits & 0x00FF0000 ) != 0x00FF0000 ) + { + prefix_byte |= (1<<3); + } + + if ( ( ack_bits & 0xFF000000 ) != 0xFF000000 ) + { + prefix_byte |= (1<<4); + } + + int sequence_difference = sequence - ack; + if ( sequence_difference < 0 ) + sequence_difference += 65536; + if ( sequence_difference <= 255 ) + prefix_byte |= (1<<5); + + reliable_write_uint8( &p, prefix_byte ); + + reliable_write_uint16( &p, sequence ); + + if ( sequence_difference <= 255 ) + { + reliable_write_uint8( &p, (uint8_t) sequence_difference ); + } + else + { + reliable_write_uint16( &p, ack ); + } + + if ( ( ack_bits & 0x000000FF ) != 0x000000FF ) + { + reliable_write_uint8( &p, (uint8_t) ( ack_bits & 0x000000FF ) ); + } + + if ( ( ack_bits & 0x0000FF00 ) != 0x0000FF00 ) + { + reliable_write_uint8( &p, (uint8_t) ( ( ack_bits & 0x0000FF00 ) >> 8 ) ); + } + + if ( ( ack_bits & 0x00FF0000 ) != 0x00FF0000 ) + { + reliable_write_uint8( &p, (uint8_t) ( ( ack_bits & 0x00FF0000 ) >> 16 ) ); + } + + if ( ( ack_bits & 0xFF000000 ) != 0xFF000000 ) + { + reliable_write_uint8( &p, (uint8_t) ( ( ack_bits & 0xFF000000 ) >> 24 ) ); + } + + reliable_assert( p - packet_data <= RELIABLE_MAX_PACKET_HEADER_BYTES ); + + return (int) ( p - packet_data ); +} + +void reliable_endpoint_send_packet( struct reliable_endpoint_t * endpoint, uint8_t * packet_data, int packet_bytes ) +{ + reliable_assert( endpoint ); + reliable_assert( packet_data ); + reliable_assert( packet_bytes > 0 ); + + if ( packet_bytes > endpoint->config.max_packet_size ) + { + reliable_printf( RELIABLE_LOG_LEVEL_ERROR, "[%s] packet too large to send. packet is %d bytes, maximum is %d\n", + endpoint->config.name, packet_bytes, endpoint->config.max_packet_size ); + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_TOO_LARGE_TO_SEND]++; + return; + } + + uint16_t sequence = endpoint->sequence++; + uint16_t ack; + uint32_t ack_bits; + + reliable_sequence_buffer_generate_ack_bits( endpoint->received_packets, &ack, &ack_bits ); + + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] sending packet %d\n", endpoint->config.name, sequence ); + + struct reliable_sent_packet_data_t * sent_packet_data = (struct reliable_sent_packet_data_t*) reliable_sequence_buffer_insert( endpoint->sent_packets, sequence ); + + reliable_assert( sent_packet_data ); + + sent_packet_data->time = endpoint->time; + sent_packet_data->packet_bytes = endpoint->config.packet_header_size + packet_bytes; + sent_packet_data->acked = 0; + + if ( packet_bytes <= endpoint->config.fragment_above ) + { + // regular packet + + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] sending packet %d without fragmentation\n", endpoint->config.name, sequence ); + + uint8_t * transmit_packet_data = (uint8_t*) endpoint->allocate_function( endpoint->allocator_context, packet_bytes + RELIABLE_MAX_PACKET_HEADER_BYTES ); + + int packet_header_bytes = reliable_write_packet_header( transmit_packet_data, sequence, ack, ack_bits ); + + memcpy( transmit_packet_data + packet_header_bytes, packet_data, packet_bytes ); + + endpoint->config.transmit_packet_function( endpoint->config.context, endpoint->config.index, sequence, transmit_packet_data, packet_header_bytes + packet_bytes ); + + endpoint->free_function( endpoint->allocator_context, transmit_packet_data ); + } + else + { + // fragmented packet + + uint8_t packet_header[RELIABLE_MAX_PACKET_HEADER_BYTES]; + + memset( packet_header, 0, RELIABLE_MAX_PACKET_HEADER_BYTES ); + + int packet_header_bytes = reliable_write_packet_header( packet_header, sequence, ack, ack_bits ); + + int num_fragments = ( packet_bytes / endpoint->config.fragment_size ) + ( ( packet_bytes % endpoint->config.fragment_size ) != 0 ? 1 : 0 ); + + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] sending packet %d as %d fragments\n", endpoint->config.name, sequence, num_fragments ); + + reliable_assert( num_fragments >= 1 ); + reliable_assert( num_fragments <= endpoint->config.max_fragments ); + + int fragment_buffer_size = RELIABLE_FRAGMENT_HEADER_BYTES + RELIABLE_MAX_PACKET_HEADER_BYTES + endpoint->config.fragment_size; + + uint8_t * fragment_packet_data = (uint8_t*) endpoint->allocate_function( endpoint->allocator_context, fragment_buffer_size ); + + uint8_t * q = packet_data; + + uint8_t * end = q + packet_bytes; + + int fragment_id; + for ( fragment_id = 0; fragment_id < num_fragments; ++fragment_id ) + { + uint8_t * p = fragment_packet_data; + + reliable_write_uint8( &p, 1 ); + reliable_write_uint16( &p, sequence ); + reliable_write_uint8( &p, (uint8_t) fragment_id ); + reliable_write_uint8( &p, (uint8_t) ( num_fragments - 1 ) ); + + if ( fragment_id == 0 ) + { + memcpy( p, packet_header, packet_header_bytes ); + p += packet_header_bytes; + } + + int bytes_to_copy = endpoint->config.fragment_size; + if ( q + bytes_to_copy > end ) + { + bytes_to_copy = (int) ( end - q ); + } + + memcpy( p, q, bytes_to_copy ); + + p += bytes_to_copy; + q += bytes_to_copy; + + int fragment_packet_bytes = (int) ( p - fragment_packet_data ); + + endpoint->config.transmit_packet_function( endpoint->config.context, endpoint->config.index, sequence, fragment_packet_data, fragment_packet_bytes ); + + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_FRAGMENTS_SENT]++; + } + + endpoint->free_function( endpoint->allocator_context, fragment_packet_data ); + } + + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_SENT]++; +} + +int reliable_read_packet_header( RELIABLE_CONST char * name, uint8_t * packet_data, int packet_bytes, uint16_t * sequence, uint16_t * ack, uint32_t * ack_bits ) +{ + if ( packet_bytes < 3 ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] packet too small for packet header (1)\n", name ); + return -1; + } + + uint8_t * p = packet_data; + + uint8_t prefix_byte = reliable_read_uint8( &p ); + + if ( ( prefix_byte & 1 ) != 0 ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] prefix byte does not indicate a regular packet\n", name ); + return -1; + } + + *sequence = reliable_read_uint16( &p ); + + if ( prefix_byte & (1<<5) ) + { + if ( packet_bytes < 3 + 1 ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] packet too small for packet header (2)\n", name ); + return -1; + } + uint8_t sequence_difference = reliable_read_uint8( &p ); + *ack = *sequence - sequence_difference; + } + else + { + if ( packet_bytes < 3 + 2 ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] packet too small for packet header (3)\n", name ); + return -1; + } + *ack = reliable_read_uint16( &p ); + } + + int expected_bytes = 0; + int i; + for ( i = 1; i <= 4; ++i ) + { + if ( prefix_byte & (1< max_fragments ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] num fragments %d outside of range of max fragments %d\n", name, *num_fragments, max_fragments ); + return -1; + } + + if ( *fragment_id >= *num_fragments ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] fragment id %d outside of range of num fragments %d\n", name, *fragment_id, *num_fragments ); + return -1; + } + + *fragment_bytes = packet_bytes - RELIABLE_FRAGMENT_HEADER_BYTES; + + uint16_t packet_sequence = 0; + uint16_t packet_ack = 0; + uint32_t packet_ack_bits = 0; + + if ( *fragment_id == 0 ) + { + int packet_header_bytes = reliable_read_packet_header( name, + packet_data + RELIABLE_FRAGMENT_HEADER_BYTES, + packet_bytes, + &packet_sequence, + &packet_ack, + &packet_ack_bits ); + + if ( packet_header_bytes < 0 ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] bad packet header in fragment\n", name ); + return -1; + } + + if ( packet_sequence != *sequence ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] bad packet sequence in fragment. expected %d, got %d\n", name, *sequence, packet_sequence ); + return -1; + } + + *fragment_bytes = packet_bytes - packet_header_bytes - RELIABLE_FRAGMENT_HEADER_BYTES; + } + + *ack = packet_ack; + *ack_bits = packet_ack_bits; + + if ( *fragment_bytes > fragment_size ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] fragment bytes %d > fragment size %d\n", name, *fragment_bytes, fragment_size ); + return - 1; + } + + if ( *fragment_id != *num_fragments - 1 && *fragment_bytes != fragment_size ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] fragment %d is %d bytes, which is not the expected fragment size %d\n", + name, *fragment_id, *fragment_bytes, fragment_size ); + return -1; + } + + return (int) ( p - packet_data ); +} + +void reliable_store_fragment_data( struct reliable_fragment_reassembly_data_t * reassembly_data, + uint16_t sequence, + uint16_t ack, + uint32_t ack_bits, + int fragment_id, + int fragment_size, + uint8_t * fragment_data, + int fragment_bytes ) +{ + if ( fragment_id == 0 ) + { + uint8_t packet_header[RELIABLE_MAX_PACKET_HEADER_BYTES]; + + memset( packet_header, 0, RELIABLE_MAX_PACKET_HEADER_BYTES ); + + reassembly_data->packet_header_bytes = reliable_write_packet_header( packet_header, sequence, ack, ack_bits ); + + memcpy( reassembly_data->packet_data + RELIABLE_MAX_PACKET_HEADER_BYTES - reassembly_data->packet_header_bytes, + packet_header, + reassembly_data->packet_header_bytes ); + + fragment_data += reassembly_data->packet_header_bytes; + fragment_bytes -= reassembly_data->packet_header_bytes; + } + + if ( fragment_id == reassembly_data->num_fragments_total - 1 ) + { + reassembly_data->packet_bytes = ( reassembly_data->num_fragments_total - 1 ) * fragment_size + fragment_bytes; + } + + memcpy( reassembly_data->packet_data + RELIABLE_MAX_PACKET_HEADER_BYTES + fragment_id * fragment_size, fragment_data, fragment_bytes ); +} + +void reliable_endpoint_receive_packet( struct reliable_endpoint_t * endpoint, uint8_t * packet_data, int packet_bytes ) +{ + reliable_assert( endpoint ); + reliable_assert( packet_data ); + reliable_assert( packet_bytes > 0 ); + + if ( packet_bytes > endpoint->config.max_packet_size + RELIABLE_MAX_PACKET_HEADER_BYTES + RELIABLE_FRAGMENT_HEADER_BYTES ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] packet too large to receive. packet is at least %d bytes, maximum is %d\n", + endpoint->config.name, packet_bytes - ( RELIABLE_MAX_PACKET_HEADER_BYTES + RELIABLE_FRAGMENT_HEADER_BYTES ), endpoint->config.max_packet_size ); + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_TOO_LARGE_TO_RECEIVE]++; + return; + } + + uint8_t prefix_byte = packet_data[0]; + + if ( ( prefix_byte & 1 ) == 0 ) + { + // regular packet + + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_RECEIVED]++; + + uint16_t sequence; + uint16_t ack; + uint32_t ack_bits; + + int packet_header_bytes = reliable_read_packet_header( endpoint->config.name, packet_data, packet_bytes, &sequence, &ack, &ack_bits ); + if ( packet_header_bytes < 0 ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] ignoring invalid packet. could not read packet header\n", endpoint->config.name ); + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_INVALID]++; + return; + } + + reliable_assert( packet_header_bytes <= packet_bytes ); + + int packet_payload_bytes = packet_bytes - packet_header_bytes; + + if ( packet_payload_bytes > endpoint->config.max_packet_size ) + { + reliable_printf( RELIABLE_LOG_LEVEL_ERROR, "[%s] packet too large to receive. packet is at %d bytes, maximum is %d\n", + endpoint->config.name, packet_payload_bytes, endpoint->config.max_packet_size ); + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_TOO_LARGE_TO_RECEIVE]++; + return; + } + + if ( !reliable_sequence_buffer_test_insert( endpoint->received_packets, sequence ) ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] ignoring stale packet %d\n", endpoint->config.name, sequence ); + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_STALE]++; + return; + } + + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] processing packet %d\n", endpoint->config.name, sequence ); + + if ( endpoint->config.process_packet_function( endpoint->config.context, + endpoint->config.index, + sequence, + packet_data + packet_header_bytes, + packet_bytes - packet_header_bytes ) ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] process packet %d successful\n", endpoint->config.name, sequence ); + + struct reliable_received_packet_data_t * received_packet_data = (struct reliable_received_packet_data_t*) + reliable_sequence_buffer_insert( endpoint->received_packets, sequence ); + + reliable_sequence_buffer_advance_with_cleanup( endpoint->fragment_reassembly, sequence, reliable_fragment_reassembly_data_cleanup ); + + reliable_assert( received_packet_data ); + + received_packet_data->time = endpoint->time; + received_packet_data->packet_bytes = endpoint->config.packet_header_size + packet_bytes; + + int i; + for ( i = 0; i < 32; ++i ) + { + if ( ack_bits & 1 ) + { + uint16_t ack_sequence = ack - ((uint16_t)i); + + struct reliable_sent_packet_data_t * sent_packet_data = (struct reliable_sent_packet_data_t*) + reliable_sequence_buffer_find( endpoint->sent_packets, ack_sequence ); + + if ( sent_packet_data && !sent_packet_data->acked && endpoint->num_acks < endpoint->config.ack_buffer_size ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] acked packet %d\n", endpoint->config.name, ack_sequence ); + endpoint->acks[endpoint->num_acks++] = ack_sequence; + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_ACKED]++; + sent_packet_data->acked = 1; + + float rtt = (float) ( endpoint->time - sent_packet_data->time ) * 1000.0f; + reliable_assert( rtt >= 0.0 ); + if ( ( endpoint->rtt == 0.0f && rtt > 0.0f ) || fabs( endpoint->rtt - rtt ) < 0.00001 ) + { + endpoint->rtt = rtt; + } + else + { + endpoint->rtt += ( rtt - endpoint->rtt ) * endpoint->config.rtt_smoothing_factor; + } + } + } + ack_bits >>= 1; + } + } + else + { + reliable_printf( RELIABLE_LOG_LEVEL_ERROR, "[%s] process packet failed\n", endpoint->config.name ); + } + } + else + { + // fragment packet + + int fragment_id; + int num_fragments; + int fragment_bytes; + + uint16_t sequence; + uint16_t ack; + uint32_t ack_bits; + + int fragment_header_bytes = reliable_read_fragment_header( endpoint->config.name, + packet_data, + packet_bytes, + endpoint->config.max_fragments, + endpoint->config.fragment_size, + &fragment_id, + &num_fragments, + &fragment_bytes, + &sequence, + &ack, + &ack_bits ); + + if ( fragment_header_bytes < 0 ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] ignoring invalid fragment. could not read fragment header\n", endpoint->config.name ); + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_FRAGMENTS_INVALID]++; + return; + } + + struct reliable_fragment_reassembly_data_t * reassembly_data = (struct reliable_fragment_reassembly_data_t*) + reliable_sequence_buffer_find( endpoint->fragment_reassembly, sequence ); + + if ( !reassembly_data ) + { + reassembly_data = (struct reliable_fragment_reassembly_data_t*) + reliable_sequence_buffer_insert_with_cleanup( endpoint->fragment_reassembly, sequence, reliable_fragment_reassembly_data_cleanup ); + + if ( !reassembly_data ) + { + reliable_printf( RELIABLE_LOG_LEVEL_ERROR, "[%s] ignoring invalid fragment. could not insert in reassembly buffer (stale)\n", endpoint->config.name ); + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_FRAGMENTS_INVALID]++; + return; + } + + reliable_sequence_buffer_advance( endpoint->received_packets, sequence ); + + int packet_buffer_size = RELIABLE_MAX_PACKET_HEADER_BYTES + num_fragments * endpoint->config.fragment_size; + + reassembly_data->sequence = sequence; + reassembly_data->ack = 0; + reassembly_data->ack_bits = 0; + reassembly_data->num_fragments_received = 0; + reassembly_data->num_fragments_total = num_fragments; + reassembly_data->packet_data = (uint8_t*) endpoint->allocate_function( endpoint->allocator_context, packet_buffer_size ); + reassembly_data->packet_bytes = 0; + memset( reassembly_data->fragment_received, 0, sizeof( reassembly_data->fragment_received ) ); + } + + if ( num_fragments != (int) reassembly_data->num_fragments_total ) + { + reliable_printf( RELIABLE_LOG_LEVEL_ERROR, "[%s] ignoring invalid fragment. fragment count mismatch. expected %d, got %d\n", + endpoint->config.name, (int) reassembly_data->num_fragments_total, num_fragments ); + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_FRAGMENTS_INVALID]++; + return; + } + + if ( reassembly_data->fragment_received[fragment_id] ) + { + reliable_printf( RELIABLE_LOG_LEVEL_ERROR, "[%s] ignoring fragment %d of packet %d. fragment already received\n", + endpoint->config.name, fragment_id, sequence ); + return; + } + + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] received fragment %d of packet %d (%d/%d)\n", + endpoint->config.name, fragment_id, sequence, reassembly_data->num_fragments_received+1, num_fragments ); + + reassembly_data->num_fragments_received++; + reassembly_data->fragment_received[fragment_id] = 1; + + reliable_store_fragment_data( reassembly_data, + sequence, + ack, + ack_bits, + fragment_id, + endpoint->config.fragment_size, + packet_data + fragment_header_bytes, + packet_bytes - fragment_header_bytes ); + + if ( reassembly_data->num_fragments_received == reassembly_data->num_fragments_total ) + { + reliable_printf( RELIABLE_LOG_LEVEL_DEBUG, "[%s] completed reassembly of packet %d\n", endpoint->config.name, sequence ); + + reliable_endpoint_receive_packet( endpoint, + reassembly_data->packet_data + RELIABLE_MAX_PACKET_HEADER_BYTES - reassembly_data->packet_header_bytes, + reassembly_data->packet_header_bytes + reassembly_data->packet_bytes ); + + reliable_sequence_buffer_remove_with_cleanup( endpoint->fragment_reassembly, sequence, reliable_fragment_reassembly_data_cleanup ); + } + + endpoint->counters[RELIABLE_ENDPOINT_COUNTER_NUM_FRAGMENTS_RECEIVED]++; + } +} + +void reliable_endpoint_free_packet( struct reliable_endpoint_t * endpoint, void * packet ) +{ + reliable_assert( endpoint ); + reliable_assert( packet ); + endpoint->free_function( endpoint->allocator_context, packet ); +} + +uint16_t * reliable_endpoint_get_acks( struct reliable_endpoint_t * endpoint, int * num_acks ) +{ + reliable_assert( endpoint ); + reliable_assert( num_acks ); + *num_acks = endpoint->num_acks; + return endpoint->acks; +} + +void reliable_endpoint_clear_acks( struct reliable_endpoint_t * endpoint ) +{ + reliable_assert( endpoint ); + endpoint->num_acks = 0; +} + +void reliable_endpoint_reset( struct reliable_endpoint_t * endpoint ) +{ + reliable_assert( endpoint ); + + endpoint->num_acks = 0; + endpoint->sequence = 0; + + memset( endpoint->acks, 0, endpoint->config.ack_buffer_size * sizeof( uint16_t ) ); + memset( endpoint->counters, 0, RELIABLE_ENDPOINT_NUM_COUNTERS * sizeof( uint64_t ) ); + + int i; + for ( i = 0; i < endpoint->config.fragment_reassembly_buffer_size; ++i ) + { + struct reliable_fragment_reassembly_data_t * reassembly_data = (struct reliable_fragment_reassembly_data_t*) + reliable_sequence_buffer_at_index( endpoint->fragment_reassembly, i ); + + if ( reassembly_data && reassembly_data->packet_data ) + { + endpoint->free_function( endpoint->allocator_context, reassembly_data->packet_data ); + reassembly_data->packet_data = NULL; + } + } + + reliable_sequence_buffer_reset( endpoint->sent_packets ); + reliable_sequence_buffer_reset( endpoint->received_packets ); + reliable_sequence_buffer_reset( endpoint->fragment_reassembly ); +} + +void reliable_endpoint_update( struct reliable_endpoint_t * endpoint, double time ) +{ + reliable_assert( endpoint ); + + endpoint->time = time; + + // calculate packet loss + { + uint32_t base_sequence = ( endpoint->sent_packets->sequence - endpoint->config.sent_packets_buffer_size + 1 ) + 0xFFFF; + int i; + int num_dropped = 0; + int num_samples = endpoint->config.sent_packets_buffer_size / 2; + for ( i = 0; i < num_samples; ++i ) + { + uint16_t sequence = (uint16_t) ( base_sequence + i ); + struct reliable_sent_packet_data_t * sent_packet_data = (struct reliable_sent_packet_data_t*) + reliable_sequence_buffer_find( endpoint->sent_packets, sequence ); + if ( sent_packet_data && !sent_packet_data->acked ) + { + num_dropped++; + } + } + float packet_loss = ( (float) num_dropped ) / ( (float) num_samples ) * 100.0f; + if ( fabs( endpoint->packet_loss - packet_loss ) > 0.00001 ) + { + endpoint->packet_loss += ( packet_loss - endpoint->packet_loss ) * endpoint->config.packet_loss_smoothing_factor; + } + else + { + endpoint->packet_loss = packet_loss; + } + } + + // calculate sent bandwidth + { + uint32_t base_sequence = ( endpoint->sent_packets->sequence - endpoint->config.sent_packets_buffer_size + 1 ) + 0xFFFF; + int i; + int bytes_sent = 0; + double start_time = FLT_MAX; + double finish_time = 0.0; + int num_samples = endpoint->config.sent_packets_buffer_size / 2; + for ( i = 0; i < num_samples; ++i ) + { + uint16_t sequence = (uint16_t) ( base_sequence + i ); + struct reliable_sent_packet_data_t * sent_packet_data = (struct reliable_sent_packet_data_t*) + reliable_sequence_buffer_find( endpoint->sent_packets, sequence ); + if ( !sent_packet_data ) + { + continue; + } + bytes_sent += sent_packet_data->packet_bytes; + if ( sent_packet_data->time < start_time ) + { + start_time = sent_packet_data->time; + } + if ( sent_packet_data->time > finish_time ) + { + finish_time = sent_packet_data->time; + } + } + if ( start_time != FLT_MAX && finish_time != 0.0 ) + { + float sent_bandwidth_kbps = (float) ( ( (double) bytes_sent ) / ( finish_time - start_time ) * 8.0f / 1000.0f ); + if ( fabs( endpoint->sent_bandwidth_kbps - sent_bandwidth_kbps ) > 0.00001 ) + { + endpoint->sent_bandwidth_kbps += ( sent_bandwidth_kbps - endpoint->sent_bandwidth_kbps ) * endpoint->config.bandwidth_smoothing_factor; + } + else + { + endpoint->sent_bandwidth_kbps = sent_bandwidth_kbps; + } + } + } + + // calculate received bandwidth + { + uint32_t base_sequence = ( endpoint->received_packets->sequence - endpoint->config.received_packets_buffer_size + 1 ) + 0xFFFF; + int i; + int bytes_sent = 0; + double start_time = FLT_MAX; + double finish_time = 0.0; + int num_samples = endpoint->config.received_packets_buffer_size / 2; + for ( i = 0; i < num_samples; ++i ) + { + uint16_t sequence = (uint16_t) ( base_sequence + i ); + struct reliable_received_packet_data_t * received_packet_data = (struct reliable_received_packet_data_t*) + reliable_sequence_buffer_find( endpoint->received_packets, sequence ); + if ( !received_packet_data ) + { + continue; + } + bytes_sent += received_packet_data->packet_bytes; + if ( received_packet_data->time < start_time ) + { + start_time = received_packet_data->time; + } + if ( received_packet_data->time > finish_time ) + { + finish_time = received_packet_data->time; + } + } + if ( start_time != FLT_MAX && finish_time != 0.0 ) + { + float received_bandwidth_kbps = (float) ( ( (double) bytes_sent ) / ( finish_time - start_time ) * 8.0f / 1000.0f ); + if ( fabs( endpoint->received_bandwidth_kbps - received_bandwidth_kbps ) > 0.00001 ) + { + endpoint->received_bandwidth_kbps += ( received_bandwidth_kbps - endpoint->received_bandwidth_kbps ) * endpoint->config.bandwidth_smoothing_factor; + } + else + { + endpoint->received_bandwidth_kbps = received_bandwidth_kbps; + } + } + } + + // calculate acked bandwidth + { + uint32_t base_sequence = ( endpoint->sent_packets->sequence - endpoint->config.sent_packets_buffer_size + 1 ) + 0xFFFF; + int i; + int bytes_sent = 0; + double start_time = FLT_MAX; + double finish_time = 0.0; + int num_samples = endpoint->config.sent_packets_buffer_size / 2; + for ( i = 0; i < num_samples; ++i ) + { + uint16_t sequence = (uint16_t) ( base_sequence + i ); + struct reliable_sent_packet_data_t * sent_packet_data = (struct reliable_sent_packet_data_t*) + reliable_sequence_buffer_find( endpoint->sent_packets, sequence ); + if ( !sent_packet_data || !sent_packet_data->acked ) + { + continue; + } + bytes_sent += sent_packet_data->packet_bytes; + if ( sent_packet_data->time < start_time ) + { + start_time = sent_packet_data->time; + } + if ( sent_packet_data->time > finish_time ) + { + finish_time = sent_packet_data->time; + } + } + if ( start_time != FLT_MAX && finish_time != 0.0 ) + { + float acked_bandwidth_kbps = (float) ( ( (double) bytes_sent ) / ( finish_time - start_time ) * 8.0f / 1000.0f ); + if ( fabs( endpoint->acked_bandwidth_kbps - acked_bandwidth_kbps ) > 0.00001 ) + { + endpoint->acked_bandwidth_kbps += ( acked_bandwidth_kbps - endpoint->acked_bandwidth_kbps ) * endpoint->config.bandwidth_smoothing_factor; + } + else + { + endpoint->acked_bandwidth_kbps = acked_bandwidth_kbps; + } + } + } +} + +float reliable_endpoint_rtt( struct reliable_endpoint_t * endpoint ) +{ + reliable_assert( endpoint ); + return endpoint->rtt; +} + +float reliable_endpoint_packet_loss( struct reliable_endpoint_t * endpoint ) +{ + reliable_assert( endpoint ); + return endpoint->packet_loss; +} + +void reliable_endpoint_bandwidth( struct reliable_endpoint_t * endpoint, float * sent_bandwidth_kbps, float * received_bandwidth_kbps, float * acked_bandwidth_kbps ) +{ + reliable_assert( endpoint ); + reliable_assert( sent_bandwidth_kbps ); + reliable_assert( acked_bandwidth_kbps ); + reliable_assert( received_bandwidth_kbps ); + *sent_bandwidth_kbps = endpoint->sent_bandwidth_kbps; + *received_bandwidth_kbps = endpoint->received_bandwidth_kbps; + *acked_bandwidth_kbps = endpoint->acked_bandwidth_kbps; +} + +RELIABLE_CONST uint64_t * reliable_endpoint_counters( struct reliable_endpoint_t * endpoint ) +{ + reliable_assert( endpoint ); + return endpoint->counters; +} + +// --------------------------------------------------------------- + +#if RELIABLE_ENABLE_TESTS + +#include +#include +#include + +static void check_handler( RELIABLE_CONST char * condition, + RELIABLE_CONST char * function, + RELIABLE_CONST char * file, + int line ) +{ + printf( "check failed: ( %s ), function %s, file %s, line %d\n", condition, function, file, line ); +#ifdef RELIABLE_DEBUG + #if defined( __GNUC__ ) + __builtin_trap(); + #elif defined( _MSC_VER ) + __debugbreak(); + #endif +#endif + exit( 1 ); +} + +#define check( condition ) \ +do \ +{ \ + if ( !(condition) ) \ + { \ + check_handler( #condition, (RELIABLE_CONST char*) __FUNCTION__, __FILE__, __LINE__ ); \ + } \ +} while(0) + +static void test_endian() +{ + uint32_t value = 0x11223344; + + char * bytes = (char*) &value; + +#if RELIABLE_LITTLE_ENDIAN + + check( bytes[0] == 0x44 ); + check( bytes[1] == 0x33 ); + check( bytes[2] == 0x22 ); + check( bytes[3] == 0x11 ); + +#else // #if RELIABLE_LITTLE_ENDIAN + + check( bytes[3] == 0x44 ); + check( bytes[2] == 0x33 ); + check( bytes[1] == 0x22 ); + check( bytes[0] == 0x11 ); + +#endif // #if RELIABLE_LITTLE_ENDIAN +} + +struct test_sequence_data_t +{ + uint16_t sequence; +}; + +#define TEST_SEQUENCE_BUFFER_SIZE 256 + +static void test_sequence_buffer() +{ + struct reliable_sequence_buffer_t * sequence_buffer = reliable_sequence_buffer_create( TEST_SEQUENCE_BUFFER_SIZE, + sizeof( struct test_sequence_data_t ), + NULL, + NULL, + NULL ); + + check( sequence_buffer ); + check( sequence_buffer->sequence == 0 ); + check( sequence_buffer->num_entries == TEST_SEQUENCE_BUFFER_SIZE ); + check( sequence_buffer->entry_stride == sizeof( struct test_sequence_data_t ) ); + + int i; + for ( i = 0; i < TEST_SEQUENCE_BUFFER_SIZE; ++i ) + { + check( reliable_sequence_buffer_find( sequence_buffer, ((uint16_t)i) ) == NULL ); + } + + for ( i = 0; i <= TEST_SEQUENCE_BUFFER_SIZE*4; ++i ) + { + struct test_sequence_data_t * entry = (struct test_sequence_data_t*) reliable_sequence_buffer_insert( sequence_buffer, ((uint16_t)i) ); + check( entry ); + entry->sequence = (uint16_t) i; + check( sequence_buffer->sequence == i + 1 ); + } + + for ( i = 0; i <= TEST_SEQUENCE_BUFFER_SIZE; ++i ) + { + struct test_sequence_data_t * entry = (struct test_sequence_data_t*) reliable_sequence_buffer_insert( sequence_buffer, ((uint16_t)i) ); + check( entry == NULL ); + } + + int index = TEST_SEQUENCE_BUFFER_SIZE * 4; + for ( i = 0; i < TEST_SEQUENCE_BUFFER_SIZE; ++i ) + { + struct test_sequence_data_t * entry = (struct test_sequence_data_t*) reliable_sequence_buffer_find( sequence_buffer, (uint16_t) index ); + check( entry ); + check( entry->sequence == (uint32_t) index ); + index--; + } + + reliable_sequence_buffer_reset( sequence_buffer ); + + check( sequence_buffer ); + check( sequence_buffer->sequence == 0 ); + check( sequence_buffer->num_entries == TEST_SEQUENCE_BUFFER_SIZE ); + check( sequence_buffer->entry_stride == sizeof( struct test_sequence_data_t ) ); + + for ( i = 0; i < TEST_SEQUENCE_BUFFER_SIZE; ++i ) + { + check( reliable_sequence_buffer_find( sequence_buffer, (uint16_t) i ) == NULL ); + } + + reliable_sequence_buffer_destroy( sequence_buffer ); +} + +static void test_generate_ack_bits() +{ + struct reliable_sequence_buffer_t * sequence_buffer = reliable_sequence_buffer_create( TEST_SEQUENCE_BUFFER_SIZE, + sizeof( struct test_sequence_data_t ), + NULL, + NULL, + NULL ); + + uint16_t ack = 0; + uint32_t ack_bits = 0xFFFFFFFF; + + reliable_sequence_buffer_generate_ack_bits( sequence_buffer, &ack, &ack_bits ); + check( ack == 0xFFFF ); + check( ack_bits == 0 ); + + int i; + for ( i = 0; i <= TEST_SEQUENCE_BUFFER_SIZE; ++i ) + { + reliable_sequence_buffer_insert( sequence_buffer, (uint16_t) i ); + } + + reliable_sequence_buffer_generate_ack_bits( sequence_buffer, &ack, &ack_bits ); + check( ack == TEST_SEQUENCE_BUFFER_SIZE ); + check( ack_bits == 0xFFFFFFFF ); + + reliable_sequence_buffer_reset( sequence_buffer ); + + uint16_t input_acks[] = { 1, 5, 9, 11 }; + int input_num_acks = sizeof( input_acks ) / sizeof( uint16_t ); + for ( i = 0; i < input_num_acks; ++i ) + { + reliable_sequence_buffer_insert( sequence_buffer, input_acks[i] ); + } + + reliable_sequence_buffer_generate_ack_bits( sequence_buffer, &ack, &ack_bits ); + + check( ack == 11 ); + check( ack_bits == ( 1 | (1<<(11-9)) | (1<<(11-5)) | (1<<(11-1)) ) ); + + reliable_sequence_buffer_destroy( sequence_buffer ); +} + +static void test_packet_header() +{ + uint16_t write_sequence; + uint16_t write_ack; + uint32_t write_ack_bits; + + uint16_t read_sequence; + uint16_t read_ack; + uint32_t read_ack_bits; + + uint8_t packet_data[RELIABLE_MAX_PACKET_HEADER_BYTES]; + + // worst case, sequence and ack are far apart, no packets acked. + + write_sequence = 10000; + write_ack = 100; + write_ack_bits = 0; + + int bytes_written = reliable_write_packet_header( packet_data, write_sequence, write_ack, write_ack_bits ); + + check( bytes_written == RELIABLE_MAX_PACKET_HEADER_BYTES ); + + int bytes_read = reliable_read_packet_header( "test_packet_header", packet_data, bytes_written, &read_sequence, &read_ack, &read_ack_bits ); + + check( bytes_read == bytes_written ); + + check( read_sequence == write_sequence ); + check( read_ack == write_ack ); + check( read_ack_bits == write_ack_bits ); + + // rare case. sequence and ack are far apart, significant # of acks are missing + + write_sequence = 10000; + write_ack = 100; + write_ack_bits = 0xFEFEFFFE; + + bytes_written = reliable_write_packet_header( packet_data, write_sequence, write_ack, write_ack_bits ); + + check( bytes_written == 1 + 2 + 2 + 3 ); + + bytes_read = reliable_read_packet_header( "test_packet_header", packet_data, bytes_written, &read_sequence, &read_ack, &read_ack_bits ); + + check( bytes_read == bytes_written ); + + check( read_sequence == write_sequence ); + check( read_ack == write_ack ); + check( read_ack_bits == write_ack_bits ); + + // common case under packet loss. sequence and ack are close together, some acks are missing + + write_sequence = 200; + write_ack = 100; + write_ack_bits = 0xFFFEFFFF; + + bytes_written = reliable_write_packet_header( packet_data, write_sequence, write_ack, write_ack_bits ); + + check( bytes_written == 1 + 2 + 1 + 1 ); + + bytes_read = reliable_read_packet_header( "test_packet_header", packet_data, bytes_written, &read_sequence, &read_ack, &read_ack_bits ); + + check( bytes_read == bytes_written ); + + check( read_sequence == write_sequence ); + check( read_ack == write_ack ); + check( read_ack_bits == write_ack_bits ); + + // ideal case. no packet loss. + + write_sequence = 200; + write_ack = 100; + write_ack_bits = 0xFFFFFFFF; + + bytes_written = reliable_write_packet_header( packet_data, write_sequence, write_ack, write_ack_bits ); + + check( bytes_written == 1 + 2 + 1 ); + + bytes_read = reliable_read_packet_header( "test_packet_header", packet_data, bytes_written, &read_sequence, &read_ack, &read_ack_bits ); + + check( bytes_read == bytes_written ); + + check( read_sequence == write_sequence ); + check( read_ack == write_ack ); + check( read_ack_bits == write_ack_bits ); +} + +struct test_context_t +{ + int drop; + int allow_packets; + struct reliable_endpoint_t * sender; + struct reliable_endpoint_t * receiver; +}; + +void test_default_context( struct test_context_t * context ) +{ + memset( context, 0, sizeof( *context ) ); + context->allow_packets = -1; +} + +static void test_transmit_packet_function( void * _context, int index, uint16_t sequence, uint8_t * packet_data, int packet_bytes ) +{ + (void) sequence; + + struct test_context_t * context = (struct test_context_t*) _context; + + if ( context->drop ) + { + return; + } + + if ( context->allow_packets >= 0 ) + { + if ( context->allow_packets == 0 ) + { + return; + } + + context->allow_packets--; + } + + if ( index == 0 ) + { + reliable_endpoint_receive_packet( context->receiver, packet_data, packet_bytes ); + } + else if ( index == 1 ) + { + reliable_endpoint_receive_packet( context->sender, packet_data, packet_bytes ); + } +} + +static int test_process_packet_function( void * _context, int index, uint16_t sequence, uint8_t * packet_data, int packet_bytes ) +{ + struct test_context_t * context = (struct test_context_t*) _context; + + (void) context; + (void) index; + (void) sequence; + (void) packet_data; + (void) packet_bytes; + + return 1; +} + +#define TEST_ACKS_NUM_ITERATIONS 256 + +static void test_acks() +{ + double time = 100.0; + + struct test_context_t context; + test_default_context( &context ); + + struct reliable_config_t sender_config; + struct reliable_config_t receiver_config; + + reliable_default_config( &sender_config ); + reliable_default_config( &receiver_config ); + + sender_config.context = &context; + sender_config.index = 0; + sender_config.transmit_packet_function = &test_transmit_packet_function; + sender_config.process_packet_function = &test_process_packet_function; + + receiver_config.context = &context; + receiver_config.index = 1; + receiver_config.transmit_packet_function = &test_transmit_packet_function; + receiver_config.process_packet_function = &test_process_packet_function; + + context.sender = reliable_endpoint_create( &sender_config, time ); + context.receiver = reliable_endpoint_create( &receiver_config, time ); + + double delta_time = 0.01; + + int i; + for ( i = 0; i < TEST_ACKS_NUM_ITERATIONS; ++i ) + { + uint8_t dummy_packet[8]; + memset( dummy_packet, 0, sizeof( dummy_packet ) ); + + reliable_endpoint_send_packet( context.sender, dummy_packet, sizeof( dummy_packet ) ); + reliable_endpoint_send_packet( context.receiver, dummy_packet, sizeof( dummy_packet ) ); + + reliable_endpoint_update( context.sender, time ); + reliable_endpoint_update( context.receiver, time ); + + time += delta_time; + } + + uint8_t sender_acked_packet[TEST_ACKS_NUM_ITERATIONS]; + memset( sender_acked_packet, 0, sizeof( sender_acked_packet ) ); + int sender_num_acks; + uint16_t * sender_acks = reliable_endpoint_get_acks( context.sender, &sender_num_acks ); + for ( i = 0; i < sender_num_acks; ++i ) + { + if ( sender_acks[i] < TEST_ACKS_NUM_ITERATIONS ) + { + sender_acked_packet[sender_acks[i]] = 1; + } + } + for ( i = 0; i < TEST_ACKS_NUM_ITERATIONS / 2; ++i ) + { + check( sender_acked_packet[i] == 1 ); + } + + uint8_t receiver_acked_packet[TEST_ACKS_NUM_ITERATIONS]; + memset( receiver_acked_packet, 0, sizeof( receiver_acked_packet ) ); + int receiver_num_acks; + uint16_t * receiver_acks = reliable_endpoint_get_acks( context.receiver, &receiver_num_acks ); + for ( i = 0; i < receiver_num_acks; ++i ) + { + if ( receiver_acks[i] < TEST_ACKS_NUM_ITERATIONS ) + receiver_acked_packet[receiver_acks[i]] = 1; + } + for ( i = 0; i < TEST_ACKS_NUM_ITERATIONS / 2; ++i ) + { + check( receiver_acked_packet[i] == 1 ); + } + + reliable_endpoint_destroy( context.sender ); + reliable_endpoint_destroy( context.receiver ); +} + +static void test_acks_packet_loss() +{ + double time = 100.0; + + struct test_context_t context; + test_default_context( &context ); + + struct reliable_config_t sender_config; + struct reliable_config_t receiver_config; + + reliable_default_config( &sender_config ); + reliable_default_config( &receiver_config ); + + sender_config.context = &context; + sender_config.index = 0; + sender_config.transmit_packet_function = &test_transmit_packet_function; + sender_config.process_packet_function = &test_process_packet_function; + + receiver_config.context = &context; + receiver_config.index = 1; + receiver_config.transmit_packet_function = &test_transmit_packet_function; + receiver_config.process_packet_function = &test_process_packet_function; + + context.sender = reliable_endpoint_create( &sender_config, time ); + context.receiver = reliable_endpoint_create( &receiver_config, time ); + + const double delta_time = 0.1f; + + int i; + for ( i = 0; i < TEST_ACKS_NUM_ITERATIONS; ++i ) + { + uint8_t dummy_packet[8]; + memset( dummy_packet, 0, sizeof( dummy_packet ) ); + + context.drop = ( i % 2 ); + + reliable_endpoint_send_packet( context.sender, dummy_packet, sizeof( dummy_packet ) ); + reliable_endpoint_send_packet( context.receiver, dummy_packet, sizeof( dummy_packet ) ); + + reliable_endpoint_update( context.sender, time ); + reliable_endpoint_update( context.receiver, time ); + + time += delta_time; + } + + uint8_t sender_acked_packet[TEST_ACKS_NUM_ITERATIONS]; + memset( sender_acked_packet, 0, sizeof( sender_acked_packet ) ); + int sender_num_acks; + uint16_t * sender_acks = reliable_endpoint_get_acks( context.sender, &sender_num_acks ); + for ( i = 0; i < sender_num_acks; ++i ) + { + if ( sender_acks[i] < TEST_ACKS_NUM_ITERATIONS ) + { + sender_acked_packet[sender_acks[i]] = 1; + } + } + for ( i = 0; i < TEST_ACKS_NUM_ITERATIONS / 2; ++i ) + { + check( sender_acked_packet[i] == (i+1) % 2 ); + } + + uint8_t receiver_acked_packet[TEST_ACKS_NUM_ITERATIONS]; + memset( receiver_acked_packet, 0, sizeof( receiver_acked_packet ) ); + int receiver_num_acks; + uint16_t * receiver_acks = reliable_endpoint_get_acks( context.sender, &receiver_num_acks ); + for ( i = 0; i < receiver_num_acks; ++i ) + { + if ( receiver_acks[i] < TEST_ACKS_NUM_ITERATIONS ) + { + receiver_acked_packet[receiver_acks[i]] = 1; + } + } + for ( i = 0; i < TEST_ACKS_NUM_ITERATIONS / 2; ++i ) + { + check( receiver_acked_packet[i] == (i+1) % 2 ); + } + + reliable_endpoint_destroy( context.sender ); + reliable_endpoint_destroy( context.receiver ); +} + +#define TEST_MAX_PACKET_BYTES (4*1024) + +static void generate_packet_data_with_size( uint16_t sequence, uint8_t * packet_data, int packet_bytes ) +{ + reliable_assert( packet_bytes >= 2 ); + reliable_assert( packet_bytes <= TEST_MAX_PACKET_BYTES ); + + packet_data[0] = (uint8_t) ( sequence & 0xFF ); + packet_data[1] = (uint8_t) ( (sequence>>8) & 0xFF ); + int i; + for ( i = 2; i < packet_bytes; ++i ) + { + packet_data[i] = (uint8_t) ( ( (int)i + sequence ) % 256 ); + } +} + +static int generate_packet_data( uint16_t sequence, uint8_t * packet_data ) +{ + int packet_bytes = ( ( (int)sequence * 1023 ) % ( TEST_MAX_PACKET_BYTES - 2 ) ) + 2; + generate_packet_data_with_size( sequence, packet_data, packet_bytes ); + return packet_bytes; +} + +static void validate_packet_data( uint8_t * packet_data, int packet_bytes ) +{ + reliable_assert( packet_bytes >= 2 ); + reliable_assert( packet_bytes <= TEST_MAX_PACKET_BYTES ); + uint16_t sequence = 0; + sequence |= (uint16_t) packet_data[0]; + sequence |= ( (uint16_t) packet_data[1] ) << 8; + check( packet_bytes == ( ( (int)sequence * 1023 ) % ( TEST_MAX_PACKET_BYTES - 2 ) ) + 2 ); + int i; + for ( i = 2; i < packet_bytes; ++i ) + { + check( packet_data[i] == (uint8_t) ( ( (int)i + sequence ) % 256 ) ); + } +} + +static int test_process_packet_function_validate( void * context, int index, uint16_t sequence, uint8_t * packet_data, int packet_bytes ) +{ + reliable_assert( packet_data ); + reliable_assert( packet_bytes > 0 ); + reliable_assert( packet_bytes <= TEST_MAX_PACKET_BYTES ); + + (void) context; + (void) index; + (void) sequence; + + validate_packet_data( packet_data, packet_bytes ); + + return 1; +} + +static int generate_packet_data_large( uint8_t* packet_data ) +{ + int data_bytes = TEST_MAX_PACKET_BYTES - 2; + reliable_assert( data_bytes >= 2) ; + reliable_assert( data_bytes <= (1 << 16) ); + + packet_data[0] = (uint8_t) (data_bytes & 0xFF); + packet_data[1] = (uint8_t) ( (data_bytes >> 8) & 0xFF ); + int i; + for ( i = 2; i < data_bytes; ++i ) + { + packet_data[i] = (uint8_t) ( i % 256 ); + } + return data_bytes + 2; +} + +static int test_process_packet_function_validate_large( void * context, int index, uint16_t sequence, uint8_t * packet_data, int packet_bytes ) +{ + reliable_assert( packet_data ); + reliable_assert( packet_bytes >= 2 ); + reliable_assert( packet_bytes <= TEST_MAX_PACKET_BYTES ); + + (void)context; + (void)index; + (void)sequence; + + uint16_t data_bytes = 0; + data_bytes |= (uint16_t) packet_data[0]; + data_bytes |= ( (uint16_t) packet_data[1] ) << 8; + check( packet_bytes == data_bytes + 2 ); + int i; + for ( i = 2; i < data_bytes; ++i ) + { + check( packet_data[i] == (uint8_t) ( i % 256 ) ); + } + + return 1; +} + +void test_packets() +{ + double time = 100.0; + + struct test_context_t context; + test_default_context( &context ); + + struct reliable_config_t sender_config; + struct reliable_config_t receiver_config; + + reliable_default_config( &sender_config ); + reliable_default_config( &receiver_config ); + + sender_config.fragment_above = 500; + receiver_config.fragment_above = 500; + +#if defined(_MSC_VER) + strcpy_s( sender_config.name, sizeof( sender_config.name ), "sender" ); +#else + strcpy( sender_config.name, "sender" ); +#endif + sender_config.context = &context; + sender_config.index = 0; + sender_config.transmit_packet_function = &test_transmit_packet_function; + sender_config.process_packet_function = &test_process_packet_function_validate; + +#if defined(_MSC_VER) + strcpy_s( receiver_config.name, sizeof( receiver_config.name ), "receiver" ); +#else + strcpy( receiver_config.name, "receiver" ); +#endif + receiver_config.context = &context; + receiver_config.index = 1; + receiver_config.transmit_packet_function = &test_transmit_packet_function; + receiver_config.process_packet_function = &test_process_packet_function_validate; + + context.sender = reliable_endpoint_create( &sender_config, time ); + context.receiver = reliable_endpoint_create( &receiver_config, time ); + + double delta_time = 0.1; + + int i; + for ( i = 0; i < 16; ++i ) + { + { + uint8_t packet_data[TEST_MAX_PACKET_BYTES]; + uint16_t sequence = reliable_endpoint_next_packet_sequence( context.sender ); + int packet_bytes = generate_packet_data( sequence, packet_data ); + reliable_endpoint_send_packet( context.sender, packet_data, packet_bytes ); + } + + { + uint8_t packet_data[TEST_MAX_PACKET_BYTES]; + uint16_t sequence = reliable_endpoint_next_packet_sequence( context.sender ); + int packet_bytes = generate_packet_data( sequence, packet_data ); + reliable_endpoint_send_packet( context.sender, packet_data, packet_bytes ); + } + + reliable_endpoint_update( context.sender, time ); + reliable_endpoint_update( context.receiver, time ); + + reliable_endpoint_clear_acks( context.sender ); + reliable_endpoint_clear_acks( context.receiver ); + + time += delta_time; + } + + reliable_endpoint_destroy( context.sender ); + reliable_endpoint_destroy( context.receiver ); +} + +void test_large_packets() +{ + double time = 100.0; + + struct test_context_t context; + test_default_context( &context ); + + struct reliable_config_t sender_config; + struct reliable_config_t receiver_config; + + reliable_default_config( &sender_config ); + reliable_default_config( &receiver_config ); + + sender_config.max_packet_size = TEST_MAX_PACKET_BYTES; + receiver_config.max_packet_size = TEST_MAX_PACKET_BYTES; + + sender_config.fragment_above = TEST_MAX_PACKET_BYTES; + receiver_config.fragment_above = TEST_MAX_PACKET_BYTES; + +#if defined(_MSC_VER) + strcpy_s( sender_config.name, sizeof( sender_config.name ), "sender" ); +#else + strcpy( sender_config.name, "sender" ); +#endif + sender_config.context = &context; + sender_config.index = 0; + sender_config.transmit_packet_function = &test_transmit_packet_function; + sender_config.process_packet_function = &test_process_packet_function_validate_large; + +#if defined(_MSC_VER) + strcpy_s( receiver_config.name, sizeof( receiver_config.name ), "receiver" ); +#else + strcpy( receiver_config.name, "receiver" ); +#endif + receiver_config.context = &context; + receiver_config.index = 1; + receiver_config.transmit_packet_function = &test_transmit_packet_function; + receiver_config.process_packet_function = &test_process_packet_function_validate_large; + + context.sender = reliable_endpoint_create( &sender_config, time ); + context.receiver = reliable_endpoint_create( &receiver_config, time ); + + { + uint8_t packet_data[TEST_MAX_PACKET_BYTES]; + int packet_bytes = generate_packet_data_large( packet_data ); + check( packet_bytes == TEST_MAX_PACKET_BYTES ); + reliable_endpoint_send_packet( context.sender, packet_data, packet_bytes ); + } + + reliable_endpoint_update( context.sender, time ); + reliable_endpoint_update( context.receiver, time ); + + reliable_endpoint_clear_acks( context.sender ); + reliable_endpoint_clear_acks( context.receiver ); + + RELIABLE_CONST uint64_t * receiver_counters = reliable_endpoint_counters( context.receiver ); + check( receiver_counters[RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_TOO_LARGE_TO_RECEIVE] == 0 ); + check( receiver_counters[RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_RECEIVED] == 1 ); + + reliable_endpoint_destroy( context.sender ); + reliable_endpoint_destroy( context.receiver ); +} + +void test_sequence_buffer_rollover() +{ + double time = 100.0; + + struct test_context_t context; + test_default_context( &context ); + + struct reliable_config_t sender_config; + struct reliable_config_t receiver_config; + + reliable_default_config( &sender_config ); + reliable_default_config( &receiver_config ); + + sender_config.fragment_above = 500; + receiver_config.fragment_above = 500; + +#if defined(_MSC_VER) + strcpy_s( sender_config.name, sizeof( sender_config.name ), "sender" ); +#else + strcpy( sender_config.name, "sender" ); +#endif + sender_config.context = &context; + sender_config.index = 0; + sender_config.transmit_packet_function = &test_transmit_packet_function; + sender_config.process_packet_function = &test_process_packet_function; + +#if defined(_MSC_VER) + strcpy_s( receiver_config.name, sizeof( receiver_config.name ), "receiver" ); +#else + strcpy( receiver_config.name, "receiver" ); +#endif + receiver_config.context = &context; + receiver_config.index = 1; + receiver_config.transmit_packet_function = &test_transmit_packet_function; + receiver_config.process_packet_function = &test_process_packet_function; + + context.sender = reliable_endpoint_create( &sender_config, time ); + context.receiver = reliable_endpoint_create( &receiver_config, time ); + + int num_packets_sent = 0; + int i; + for (i = 0; i <= 32767; ++i) + { + uint8_t packet_data[16]; + int packet_bytes = sizeof( packet_data ) / sizeof( uint8_t ); + reliable_endpoint_next_packet_sequence( context.sender ); + reliable_endpoint_send_packet( context.sender, packet_data, packet_bytes ); + + ++num_packets_sent; + } + + uint8_t packet_data[TEST_MAX_PACKET_BYTES]; + int packet_bytes = sizeof( packet_data ) / sizeof( uint8_t ); + reliable_endpoint_next_packet_sequence( context.sender ); + reliable_endpoint_send_packet( context.sender, packet_data, packet_bytes ); + ++num_packets_sent; + + RELIABLE_CONST uint64_t * receiver_counters = reliable_endpoint_counters( context.receiver ); + + check( receiver_counters[RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_RECEIVED] == (uint16_t) num_packets_sent ); + check( receiver_counters[RELIABLE_ENDPOINT_COUNTER_NUM_FRAGMENTS_INVALID] == 0 ); + + reliable_endpoint_destroy( context.sender ); + reliable_endpoint_destroy( context.receiver ); +} + +#define RELIABLE_ARRAY_SIZE(x) ( int( sizeof((x)) / sizeof((x)[0]) ) ) + +struct test_tracking_allocate_context_t +{ + void* active_allocations[1024]; +}; + +void * test_tracking_allocate_function( void * context, uint64_t bytes ) +{ + struct test_tracking_allocate_context_t* tracking_context = (struct test_tracking_allocate_context_t*)context; + void * allocation = malloc( bytes ); + int tracking_index; + for ( tracking_index = 0; tracking_index < RELIABLE_ARRAY_SIZE(tracking_context->active_allocations); ++tracking_index ) + { + if ( tracking_context->active_allocations[tracking_index] == NULL ) + { + break; + } + } + + reliable_assert(tracking_index < RELIABLE_ARRAY_SIZE(tracking_context->active_allocations)); + tracking_context->active_allocations[tracking_index] = allocation; + return allocation; +} + +void test_tracking_free_function( void * context, void * pointer ) +{ + struct test_tracking_allocate_context_t* tracking_context = (struct test_tracking_allocate_context_t*)context; + int tracking_index; + for ( tracking_index = 0; tracking_index < RELIABLE_ARRAY_SIZE(tracking_context->active_allocations); ++tracking_index ) + { + if ( tracking_context->active_allocations[tracking_index] == pointer ) + { + break; + } + } + + reliable_assert(tracking_index < RELIABLE_ARRAY_SIZE(tracking_context->active_allocations)); + tracking_context->active_allocations[tracking_index] = NULL; + free( pointer ); +} + +void test_fragment_cleanup() +{ + double time = 100.0; + + struct test_context_t context; + test_default_context( &context ); + + struct test_tracking_allocate_context_t tracking_alloc_context; + memset( &tracking_alloc_context, 0, sizeof( tracking_alloc_context ) ); + + struct reliable_config_t sender_config; + struct reliable_config_t receiver_config; + + reliable_default_config( &sender_config ); + reliable_default_config( &receiver_config ); + + receiver_config.allocator_context = &tracking_alloc_context; + receiver_config.allocate_function = &test_tracking_allocate_function; + receiver_config.free_function = &test_tracking_free_function; + receiver_config.fragment_reassembly_buffer_size = 4; + +#if defined(_MSC_VER) + strcpy_s( sender_config.name, sizeof( sender_config.name ), "sender" ); +#else + strcpy( sender_config.name, "sender" ); +#endif + sender_config.context = &context; + sender_config.index = 0; + sender_config.transmit_packet_function = &test_transmit_packet_function; + sender_config.process_packet_function = &test_process_packet_function; + +#if defined(_MSC_VER) + strcpy_s( receiver_config.name, sizeof( receiver_config.name ), "receiver" ); +#else + strcpy( receiver_config.name, "receiver" ); +#endif + receiver_config.context = &context; + receiver_config.index = 1; + receiver_config.transmit_packet_function = &test_transmit_packet_function; + receiver_config.process_packet_function = &test_process_packet_function; + + context.sender = reliable_endpoint_create( &sender_config, time ); + context.receiver = reliable_endpoint_create( &receiver_config, time ); + + double delta_time = 0.1; + + int packet_sizes[] = { + sender_config.fragment_size + sender_config.fragment_size/2, + 10, + 10, + 10, + 10, + }; + + // Make sure we're sending more than receiver_config.fragment_reassembly_buffer_size packets, so the buffer wraps around. + reliable_assert( RELIABLE_ARRAY_SIZE( packet_sizes ) > receiver_config.fragment_reassembly_buffer_size ); + + int i; + for ( i = 0; i < RELIABLE_ARRAY_SIZE( packet_sizes ); ++i ) + { + // Only allow one packet per transmit, so that our fragmented packets are only partially + // delivered. + context.allow_packets = 1; + { + uint8_t packet_data[TEST_MAX_PACKET_BYTES]; + uint16_t sequence = reliable_endpoint_next_packet_sequence( context.sender ); + generate_packet_data_with_size( sequence, packet_data, packet_sizes[i] ); + reliable_endpoint_send_packet( context.sender, packet_data, packet_sizes[i]); + } + + reliable_endpoint_update( context.sender, time ); + reliable_endpoint_update( context.receiver, time ); + + reliable_endpoint_clear_acks( context.sender ); + reliable_endpoint_clear_acks( context.receiver ); + + time += delta_time; + } + + reliable_endpoint_destroy( context.sender ); + reliable_endpoint_destroy( context.receiver ); + + // Make sure that there is no memory that hasn't been freed. + int tracking_index; + for ( tracking_index = 0; tracking_index < RELIABLE_ARRAY_SIZE(tracking_alloc_context.active_allocations); ++tracking_index ) + { + check( tracking_alloc_context.active_allocations[tracking_index] == NULL ); + } +} + +#define RUN_TEST( test_function ) \ + do \ + { \ + printf( #test_function "\n" ); \ + test_function(); \ + } \ + while (0) + +void reliable_test() +{ + //while ( 1 ) + { + RUN_TEST( test_endian ); + RUN_TEST( test_sequence_buffer ); + RUN_TEST( test_generate_ack_bits ); + RUN_TEST( test_packet_header ); + RUN_TEST( test_acks ); + RUN_TEST( test_acks_packet_loss ); + RUN_TEST( test_packets ); + RUN_TEST( test_large_packets ); + RUN_TEST( test_sequence_buffer_rollover ); + RUN_TEST( test_fragment_cleanup ); + } +} + +#endif // #if RELIABLE_ENABLE_TESTS diff --git a/reliable/reliable.h b/reliable/reliable.h new file mode 100644 index 00000000..031c1ffc --- /dev/null +++ b/reliable/reliable.h @@ -0,0 +1,176 @@ +/* + reliable + + Copyright © 2017 - 2024, Mas Bandwidth LLC + + Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: + + 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer + in the documentation and/or other materials provided with the distribution. + + 3. Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived + from this software without specific prior written permission. + + THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, + INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, + SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR + SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, + WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE + USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ + +#ifndef RELIABLE_H +#define RELIABLE_H + +#include + +#if !defined(RELIABLE_DEBUG) && !defined(RELIABLE_RELEASE) +#if defined(NDEBUG) +#define RELIABLE_RELEASE +#else +#define RELIABLE_DEBUG +#endif +#elif defined(RELIABLE_DEBUG) && defined(RELIABLE_RELEASE) +#error Can only define one of debug & release +#endif + +#if defined(__386__) || defined(i386) || defined(__i386__) \ + || defined(__X86) || defined(_M_IX86) \ + || defined(_M_X64) || defined(__x86_64__) \ + || defined(alpha) || defined(__alpha) || defined(__alpha__) \ + || defined(_M_ALPHA) \ + || defined(ARM) || defined(_ARM) || defined(__arm__) \ + || defined(__aarch64__) \ + || defined(WIN32) || defined(_WIN32) || defined(__WIN32__) \ + || defined(_WIN32_WCE) || defined(__NT__) \ + || defined(__MIPSEL__) + #define RELIABLE_LITTLE_ENDIAN 1 +#else + #define RELIABLE_BIG_ENDIAN 1 +#endif + +#define RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_SENT 0 +#define RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_RECEIVED 1 +#define RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_ACKED 2 +#define RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_STALE 3 +#define RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_INVALID 4 +#define RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_TOO_LARGE_TO_SEND 5 +#define RELIABLE_ENDPOINT_COUNTER_NUM_PACKETS_TOO_LARGE_TO_RECEIVE 6 +#define RELIABLE_ENDPOINT_COUNTER_NUM_FRAGMENTS_SENT 7 +#define RELIABLE_ENDPOINT_COUNTER_NUM_FRAGMENTS_RECEIVED 8 +#define RELIABLE_ENDPOINT_COUNTER_NUM_FRAGMENTS_INVALID 9 +#define RELIABLE_ENDPOINT_NUM_COUNTERS 10 + +#define RELIABLE_MAX_PACKET_HEADER_BYTES 9 +#define RELIABLE_FRAGMENT_HEADER_BYTES 5 + +#define RELIABLE_LOG_LEVEL_NONE 0 +#define RELIABLE_LOG_LEVEL_ERROR 1 +#define RELIABLE_LOG_LEVEL_INFO 2 +#define RELIABLE_LOG_LEVEL_DEBUG 3 + +#define RELIABLE_OK 1 +#define RELIABLE_ERROR 0 + +#ifdef __cplusplus +#define RELIABLE_CONST const +extern "C" { +#else +#if defined(__STDC__) +#define RELIABLE_CONST const +#else +#define RELIABLE_CONST +#endif +#endif + +int reliable_init(void); + +void reliable_term(void); + +struct reliable_config_t +{ + char name[256]; + void * context; + int index; + int max_packet_size; + int fragment_above; + int max_fragments; + int fragment_size; + int ack_buffer_size; + int sent_packets_buffer_size; + int received_packets_buffer_size; + int fragment_reassembly_buffer_size; + float rtt_smoothing_factor; + float packet_loss_smoothing_factor; + float bandwidth_smoothing_factor; + int packet_header_size; + void (*transmit_packet_function)(void*,int,uint16_t,uint8_t*,int); + int (*process_packet_function)(void*,int,uint16_t,uint8_t*,int); + void * allocator_context; + void * (*allocate_function)(void*,uint64_t); + void (*free_function)(void*,void*); +}; + +void reliable_default_config( struct reliable_config_t * config ); + +struct reliable_endpoint_t * reliable_endpoint_create( struct reliable_config_t * config, double time ); + +uint16_t reliable_endpoint_next_packet_sequence( struct reliable_endpoint_t * endpoint ); + +void reliable_endpoint_send_packet( struct reliable_endpoint_t * endpoint, uint8_t * packet_data, int packet_bytes ); + +void reliable_endpoint_receive_packet( struct reliable_endpoint_t * endpoint, uint8_t * packet_data, int packet_bytes ); + +void reliable_endpoint_free_packet( struct reliable_endpoint_t * endpoint, void * packet ); + +uint16_t * reliable_endpoint_get_acks( struct reliable_endpoint_t * endpoint, int * num_acks ); + +void reliable_endpoint_clear_acks( struct reliable_endpoint_t * endpoint ); + +void reliable_endpoint_reset( struct reliable_endpoint_t * endpoint ); + +void reliable_endpoint_update( struct reliable_endpoint_t * endpoint, double time ); + +float reliable_endpoint_rtt( struct reliable_endpoint_t * endpoint ); + +float reliable_endpoint_packet_loss( struct reliable_endpoint_t * endpoint ); + +void reliable_endpoint_bandwidth( struct reliable_endpoint_t * endpoint, float * sent_bandwidth_kbps, float * received_bandwidth_kbps, float * acked_bandwidth_kpbs ); + +RELIABLE_CONST uint64_t * reliable_endpoint_counters( struct reliable_endpoint_t * endpoint ); + +void reliable_endpoint_destroy( struct reliable_endpoint_t * endpoint ); + +void reliable_log_level( int level ); + +void reliable_set_printf_function( int (*function)( RELIABLE_CONST char *, ... ) ); + +extern void (*reliable_assert_function)( RELIABLE_CONST char *, RELIABLE_CONST char *, RELIABLE_CONST char * file, int line ); + +#ifdef RELIABLE_DEBUG +#define reliable_assert( condition ) \ +do \ +{ \ + if ( !(condition) ) \ + { \ + reliable_assert_function( #condition, __FUNCTION__, __FILE__, __LINE__ ); \ + exit(1); \ + } \ +} while(0) +#else +#define reliable_assert( ignore ) ((void)0) +#endif + +void reliable_set_assert_function( void (*function)( RELIABLE_CONST char * /*condition*/, + RELIABLE_CONST char * /*function*/, + RELIABLE_CONST char * /*file*/, + int /*line*/ ) ); + +#ifdef __cplusplus +} +#endif + +#endif // #ifndef RELIABLE_H diff --git a/sodium/sodium.h b/sodium/sodium.h new file mode 100644 index 00000000..af635f33 --- /dev/null +++ b/sodium/sodium.h @@ -0,0 +1,48 @@ + +#ifndef sodium_H +#define sodium_H + +#include "sodium_version.h" +#include "sodium_core.h" +#include "sodium_crypto_aead_chacha20poly1305.h" +#include "sodium_crypto_aead_xchacha20poly1305.h" +#include "sodium_crypto_auth.h" +#include "sodium_crypto_auth_hmacsha512.h" +#include "sodium_crypto_auth_hmacsha512256.h" +#include "sodium_crypto_box.h" +#include "sodium_crypto_box_curve25519xsalsa20poly1305.h" +#include "sodium_crypto_core_hsalsa20.h" +#include "sodium_crypto_core_hchacha20.h" +#include "sodium_crypto_core_salsa20.h" +#include "sodium_crypto_core_salsa2012.h" +#include "sodium_crypto_core_salsa208.h" +#include "sodium_crypto_generichash.h" +#include "sodium_crypto_generichash_blake2b.h" +#include "sodium_crypto_hash.h" +#include "sodium_crypto_hash_sha512.h" +#include "sodium_crypto_kx.h" +#include "sodium_crypto_onetimeauth.h" +#include "sodium_crypto_onetimeauth_poly1305.h" +#include "sodium_crypto_scalarmult.h" +#include "sodium_crypto_scalarmult_curve25519.h" +#include "sodium_crypto_secretbox.h" +#include "sodium_crypto_secretbox_xsalsa20poly1305.h" +#include "sodium_crypto_secretstream_xchacha20poly1305.h" +#include "sodium_crypto_shorthash.h" +#include "sodium_crypto_shorthash_siphash24.h" +#include "sodium_crypto_sign.h" +#include "sodium_crypto_sign_ed25519.h" +#include "sodium_crypto_stream.h" +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_crypto_stream_salsa20.h" +#include "sodium_crypto_stream_xsalsa20.h" +#include "sodium_crypto_verify_16.h" +#include "sodium_crypto_verify_32.h" +#include "sodium_crypto_verify_64.h" +#include "sodium_randombytes.h" +#include "sodium_randombytes_salsa20_random.h" +#include "sodium_randombytes_sysrandom.h" +#include "sodium_runtime.h" +#include "sodium_utils.h" + +#endif diff --git a/sodium/sodium_aead_chacha20poly1305.c b/sodium/sodium_aead_chacha20poly1305.c new file mode 100644 index 00000000..5ef39973 --- /dev/null +++ b/sodium/sodium_aead_chacha20poly1305.c @@ -0,0 +1,400 @@ + +#include +#include +#include +#include + +#include "sodium_core.h" +#include "sodium_crypto_aead_chacha20poly1305.h" +#include "sodium_crypto_onetimeauth_poly1305.h" +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_crypto_verify_16.h" +#include "sodium_randombytes.h" +#include "sodium_utils.h" + +#include "sodium_private_chacha20_ietf_ext.h" +#include "sodium_private_common.h" + +static const unsigned char _pad0[16] = { 0 }; + +int +crypto_aead_chacha20poly1305_encrypt_detached(unsigned char *c, + unsigned char *mac, + unsigned long long *maclen_p, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *nsec, + const unsigned char *npub, + const unsigned char *k) +{ + crypto_onetimeauth_poly1305_state state; + unsigned char block0[64U]; + unsigned char slen[8U]; + + (void) nsec; + crypto_stream_chacha20(block0, sizeof block0, npub, k); + crypto_onetimeauth_poly1305_init(&state, block0); + sodium_memzero(block0, sizeof block0); + + crypto_onetimeauth_poly1305_update(&state, ad, adlen); + STORE64_LE(slen, (uint64_t) adlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + crypto_stream_chacha20_xor_ic(c, m, mlen, npub, 1U, k); + + crypto_onetimeauth_poly1305_update(&state, c, mlen); + STORE64_LE(slen, (uint64_t) mlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + crypto_onetimeauth_poly1305_final(&state, mac); + sodium_memzero(&state, sizeof state); + + if (maclen_p != NULL) { + *maclen_p = crypto_aead_chacha20poly1305_ABYTES; + } + return 0; +} + +int +crypto_aead_chacha20poly1305_encrypt(unsigned char *c, + unsigned long long *clen_p, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *nsec, + const unsigned char *npub, + const unsigned char *k) +{ + unsigned long long clen = 0ULL; + int ret; + + if (mlen > crypto_aead_chacha20poly1305_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + ret = crypto_aead_chacha20poly1305_encrypt_detached(c, + c + mlen, NULL, + m, mlen, + ad, adlen, + nsec, npub, k); + if (clen_p != NULL) { + if (ret == 0) { + clen = mlen + crypto_aead_chacha20poly1305_ABYTES; + } + *clen_p = clen; + } + return ret; +} + +int +crypto_aead_chacha20poly1305_ietf_encrypt_detached(unsigned char *c, + unsigned char *mac, + unsigned long long *maclen_p, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *nsec, + const unsigned char *npub, + const unsigned char *k) +{ + crypto_onetimeauth_poly1305_state state; + unsigned char block0[64U]; + unsigned char slen[8U]; + + (void) nsec; + crypto_stream_chacha20_ietf(block0, sizeof block0, npub, k); + crypto_onetimeauth_poly1305_init(&state, block0); + sodium_memzero(block0, sizeof block0); + + crypto_onetimeauth_poly1305_update(&state, ad, adlen); + crypto_onetimeauth_poly1305_update(&state, _pad0, (0x10 - adlen) & 0xf); + + crypto_stream_chacha20_ietf_xor_ic(c, m, mlen, npub, 1U, k); + + crypto_onetimeauth_poly1305_update(&state, c, mlen); + crypto_onetimeauth_poly1305_update(&state, _pad0, (0x10 - mlen) & 0xf); + + STORE64_LE(slen, (uint64_t) adlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + STORE64_LE(slen, (uint64_t) mlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + crypto_onetimeauth_poly1305_final(&state, mac); + sodium_memzero(&state, sizeof state); + + if (maclen_p != NULL) { + *maclen_p = crypto_aead_chacha20poly1305_ietf_ABYTES; + } + return 0; +} + +int +crypto_aead_chacha20poly1305_ietf_encrypt(unsigned char *c, + unsigned long long *clen_p, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *nsec, + const unsigned char *npub, + const unsigned char *k) +{ + unsigned long long clen = 0ULL; + int ret; + + if (mlen > crypto_aead_chacha20poly1305_ietf_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + ret = crypto_aead_chacha20poly1305_ietf_encrypt_detached(c, + c + mlen, NULL, + m, mlen, + ad, adlen, + nsec, npub, k); + if (clen_p != NULL) { + if (ret == 0) { + clen = mlen + crypto_aead_chacha20poly1305_ietf_ABYTES; + } + *clen_p = clen; + } + return ret; +} + +int +crypto_aead_chacha20poly1305_decrypt_detached(unsigned char *m, + unsigned char *nsec, + const unsigned char *c, + unsigned long long clen, + const unsigned char *mac, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *npub, + const unsigned char *k) +{ + crypto_onetimeauth_poly1305_state state; + unsigned char block0[64U]; + unsigned char slen[8U]; + unsigned char computed_mac[crypto_aead_chacha20poly1305_ABYTES]; + unsigned long long mlen; + int ret; + + (void) nsec; + crypto_stream_chacha20(block0, sizeof block0, npub, k); + crypto_onetimeauth_poly1305_init(&state, block0); + sodium_memzero(block0, sizeof block0); + + crypto_onetimeauth_poly1305_update(&state, ad, adlen); + STORE64_LE(slen, (uint64_t) adlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + mlen = clen; + crypto_onetimeauth_poly1305_update(&state, c, mlen); + STORE64_LE(slen, (uint64_t) mlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + crypto_onetimeauth_poly1305_final(&state, computed_mac); + sodium_memzero(&state, sizeof state); + + COMPILER_ASSERT(sizeof computed_mac == 16U); + ret = crypto_verify_16(computed_mac, mac); + sodium_memzero(computed_mac, sizeof computed_mac); + if (m == NULL) { + return ret; + } + if (ret != 0) { + memset(m, 0, mlen); + return -1; + } + crypto_stream_chacha20_xor_ic(m, c, mlen, npub, 1U, k); + + return 0; +} + +int +crypto_aead_chacha20poly1305_decrypt(unsigned char *m, + unsigned long long *mlen_p, + unsigned char *nsec, + const unsigned char *c, + unsigned long long clen, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *npub, + const unsigned char *k) +{ + unsigned long long mlen = 0ULL; + int ret = -1; + + if (clen >= crypto_aead_chacha20poly1305_ABYTES) { + ret = crypto_aead_chacha20poly1305_decrypt_detached + (m, nsec, + c, clen - crypto_aead_chacha20poly1305_ABYTES, + c + clen - crypto_aead_chacha20poly1305_ABYTES, + ad, adlen, npub, k); + } + if (mlen_p != NULL) { + if (ret == 0) { + mlen = clen - crypto_aead_chacha20poly1305_ABYTES; + } + *mlen_p = mlen; + } + return ret; +} + +int +crypto_aead_chacha20poly1305_ietf_decrypt_detached(unsigned char *m, + unsigned char *nsec, + const unsigned char *c, + unsigned long long clen, + const unsigned char *mac, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *npub, + const unsigned char *k) +{ + crypto_onetimeauth_poly1305_state state; + unsigned char block0[64U]; + unsigned char slen[8U]; + unsigned char computed_mac[crypto_aead_chacha20poly1305_ietf_ABYTES]; + unsigned long long mlen; + int ret; + + (void) nsec; + crypto_stream_chacha20_ietf(block0, sizeof block0, npub, k); + crypto_onetimeauth_poly1305_init(&state, block0); + sodium_memzero(block0, sizeof block0); + + crypto_onetimeauth_poly1305_update(&state, ad, adlen); + crypto_onetimeauth_poly1305_update(&state, _pad0, (0x10 - adlen) & 0xf); + + mlen = clen; + crypto_onetimeauth_poly1305_update(&state, c, mlen); + crypto_onetimeauth_poly1305_update(&state, _pad0, (0x10 - mlen) & 0xf); + + STORE64_LE(slen, (uint64_t) adlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + STORE64_LE(slen, (uint64_t) mlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + crypto_onetimeauth_poly1305_final(&state, computed_mac); + sodium_memzero(&state, sizeof state); + + COMPILER_ASSERT(sizeof computed_mac == 16U); + ret = crypto_verify_16(computed_mac, mac); + sodium_memzero(computed_mac, sizeof computed_mac); + if (m == NULL) { + return ret; + } + if (ret != 0) { + memset(m, 0, mlen); + return -1; + } + crypto_stream_chacha20_ietf_xor_ic(m, c, mlen, npub, 1U, k); + + return 0; +} + +int +crypto_aead_chacha20poly1305_ietf_decrypt(unsigned char *m, + unsigned long long *mlen_p, + unsigned char *nsec, + const unsigned char *c, + unsigned long long clen, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *npub, + const unsigned char *k) +{ + unsigned long long mlen = 0ULL; + int ret = -1; + + if (clen >= crypto_aead_chacha20poly1305_ietf_ABYTES) { + ret = crypto_aead_chacha20poly1305_ietf_decrypt_detached + (m, nsec, + c, clen - crypto_aead_chacha20poly1305_ietf_ABYTES, + c + clen - crypto_aead_chacha20poly1305_ietf_ABYTES, + ad, adlen, npub, k); + } + if (mlen_p != NULL) { + if (ret == 0) { + mlen = clen - crypto_aead_chacha20poly1305_ietf_ABYTES; + } + *mlen_p = mlen; + } + return ret; +} + +size_t +crypto_aead_chacha20poly1305_ietf_keybytes(void) +{ + return crypto_aead_chacha20poly1305_ietf_KEYBYTES; +} + +size_t +crypto_aead_chacha20poly1305_ietf_npubbytes(void) +{ + return crypto_aead_chacha20poly1305_ietf_NPUBBYTES; +} + +size_t +crypto_aead_chacha20poly1305_ietf_nsecbytes(void) +{ + return crypto_aead_chacha20poly1305_ietf_NSECBYTES; +} + +size_t +crypto_aead_chacha20poly1305_ietf_abytes(void) +{ + return crypto_aead_chacha20poly1305_ietf_ABYTES; +} + +size_t +crypto_aead_chacha20poly1305_ietf_messagebytes_max(void) +{ + return crypto_aead_chacha20poly1305_ietf_MESSAGEBYTES_MAX; +} + +void +crypto_aead_chacha20poly1305_ietf_keygen(unsigned char k[crypto_aead_chacha20poly1305_ietf_KEYBYTES]) +{ + randombytes_buf(k, crypto_aead_chacha20poly1305_ietf_KEYBYTES); +} + +size_t +crypto_aead_chacha20poly1305_keybytes(void) +{ + return crypto_aead_chacha20poly1305_KEYBYTES; +} + +size_t +crypto_aead_chacha20poly1305_npubbytes(void) +{ + return crypto_aead_chacha20poly1305_NPUBBYTES; +} + +size_t +crypto_aead_chacha20poly1305_nsecbytes(void) +{ + return crypto_aead_chacha20poly1305_NSECBYTES; +} + +size_t +crypto_aead_chacha20poly1305_abytes(void) +{ + return crypto_aead_chacha20poly1305_ABYTES; +} + +size_t +crypto_aead_chacha20poly1305_messagebytes_max(void) +{ + return crypto_aead_chacha20poly1305_MESSAGEBYTES_MAX; +} + +void +crypto_aead_chacha20poly1305_keygen(unsigned char k[crypto_aead_chacha20poly1305_KEYBYTES]) +{ + randombytes_buf(k, crypto_aead_chacha20poly1305_KEYBYTES); +} diff --git a/sodium/sodium_aead_xchacha20poly1305.c b/sodium/sodium_aead_xchacha20poly1305.c new file mode 100644 index 00000000..74b3f083 --- /dev/null +++ b/sodium/sodium_aead_xchacha20poly1305.c @@ -0,0 +1,262 @@ + +#include +#include +#include +#include + +#include "sodium_core.h" +#include "sodium_crypto_aead_chacha20poly1305.h" +#include "sodium_crypto_aead_xchacha20poly1305.h" +#include "sodium_crypto_core_hchacha20.h" +#include "sodium_crypto_onetimeauth_poly1305.h" +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_crypto_verify_16.h" +#include "sodium_randombytes.h" +#include "sodium_utils.h" + +#include "sodium_private_chacha20_ietf_ext.h" +#include "sodium_private_common.h" + +static const unsigned char _pad0[16] = { 0 }; + +static int +_encrypt_detached(unsigned char *c, + unsigned char *mac, + unsigned long long *maclen_p, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *nsec, + const unsigned char *npub, + const unsigned char *k) +{ + crypto_onetimeauth_poly1305_state state; + unsigned char block0[64U]; + unsigned char slen[8U]; + + (void) nsec; + crypto_stream_chacha20_ietf_ext(block0, sizeof block0, npub, k); + crypto_onetimeauth_poly1305_init(&state, block0); + sodium_memzero(block0, sizeof block0); + + crypto_onetimeauth_poly1305_update(&state, ad, adlen); + crypto_onetimeauth_poly1305_update(&state, _pad0, (0x10 - adlen) & 0xf); + + crypto_stream_chacha20_ietf_ext_xor_ic(c, m, mlen, npub, 1U, k); + + crypto_onetimeauth_poly1305_update(&state, c, mlen); + crypto_onetimeauth_poly1305_update(&state, _pad0, (0x10 - mlen) & 0xf); + + STORE64_LE(slen, (uint64_t) adlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + STORE64_LE(slen, (uint64_t) mlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + crypto_onetimeauth_poly1305_final(&state, mac); + sodium_memzero(&state, sizeof state); + + if (maclen_p != NULL) { + *maclen_p = crypto_aead_chacha20poly1305_ietf_ABYTES; + } + return 0; +} + +static int +_decrypt_detached(unsigned char *m, + unsigned char *nsec, + const unsigned char *c, + unsigned long long clen, + const unsigned char *mac, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *npub, + const unsigned char *k) +{ + crypto_onetimeauth_poly1305_state state; + unsigned char block0[64U]; + unsigned char slen[8U]; + unsigned char computed_mac[crypto_aead_chacha20poly1305_ietf_ABYTES]; + unsigned long long mlen; + int ret; + + (void) nsec; + crypto_stream_chacha20_ietf_ext(block0, sizeof block0, npub, k); + crypto_onetimeauth_poly1305_init(&state, block0); + sodium_memzero(block0, sizeof block0); + + crypto_onetimeauth_poly1305_update(&state, ad, adlen); + crypto_onetimeauth_poly1305_update(&state, _pad0, (0x10 - adlen) & 0xf); + + mlen = clen; + crypto_onetimeauth_poly1305_update(&state, c, mlen); + crypto_onetimeauth_poly1305_update(&state, _pad0, (0x10 - mlen) & 0xf); + + STORE64_LE(slen, (uint64_t) adlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + STORE64_LE(slen, (uint64_t) mlen); + crypto_onetimeauth_poly1305_update(&state, slen, sizeof slen); + + crypto_onetimeauth_poly1305_final(&state, computed_mac); + sodium_memzero(&state, sizeof state); + + COMPILER_ASSERT(sizeof computed_mac == 16U); + ret = crypto_verify_16(computed_mac, mac); + sodium_memzero(computed_mac, sizeof computed_mac); + if (m == NULL) { + return ret; + } + if (ret != 0) { + memset(m, 0, mlen); + return -1; + } + crypto_stream_chacha20_ietf_ext_xor_ic(m, c, mlen, npub, 1U, k); + + return 0; +} + +int +crypto_aead_xchacha20poly1305_ietf_encrypt_detached(unsigned char *c, + unsigned char *mac, + unsigned long long *maclen_p, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *nsec, + const unsigned char *npub, + const unsigned char *k) +{ + unsigned char k2[crypto_core_hchacha20_OUTPUTBYTES]; + unsigned char npub2[crypto_aead_chacha20poly1305_ietf_NPUBBYTES] = { 0 }; + int ret; + + crypto_core_hchacha20(k2, npub, k, NULL); + memcpy(npub2 + 4, npub + crypto_core_hchacha20_INPUTBYTES, + crypto_aead_chacha20poly1305_ietf_NPUBBYTES - 4); + ret = _encrypt_detached(c, mac, maclen_p, m, mlen, ad, adlen, + nsec, npub2, k2); + sodium_memzero(k2, crypto_core_hchacha20_OUTPUTBYTES); + + return ret; +} + +int +crypto_aead_xchacha20poly1305_ietf_encrypt(unsigned char *c, + unsigned long long *clen_p, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *nsec, + const unsigned char *npub, + const unsigned char *k) +{ + unsigned long long clen = 0ULL; + int ret; + + if (mlen > crypto_aead_xchacha20poly1305_ietf_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + ret = crypto_aead_xchacha20poly1305_ietf_encrypt_detached + (c, c + mlen, NULL, m, mlen, ad, adlen, nsec, npub, k); + if (clen_p != NULL) { + if (ret == 0) { + clen = mlen + crypto_aead_xchacha20poly1305_ietf_ABYTES; + } + *clen_p = clen; + } + return ret; +} + +int +crypto_aead_xchacha20poly1305_ietf_decrypt_detached(unsigned char *m, + unsigned char *nsec, + const unsigned char *c, + unsigned long long clen, + const unsigned char *mac, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *npub, + const unsigned char *k) +{ + unsigned char k2[crypto_core_hchacha20_OUTPUTBYTES]; + unsigned char npub2[crypto_aead_chacha20poly1305_ietf_NPUBBYTES] = { 0 }; + int ret; + + crypto_core_hchacha20(k2, npub, k, NULL); + memcpy(npub2 + 4, npub + crypto_core_hchacha20_INPUTBYTES, + crypto_aead_chacha20poly1305_ietf_NPUBBYTES - 4); + ret = _decrypt_detached(m, nsec, c, clen, mac, ad, adlen, npub2, k2); + sodium_memzero(k2, crypto_core_hchacha20_OUTPUTBYTES); + + return ret; +} + +int +crypto_aead_xchacha20poly1305_ietf_decrypt(unsigned char *m, + unsigned long long *mlen_p, + unsigned char *nsec, + const unsigned char *c, + unsigned long long clen, + const unsigned char *ad, + unsigned long long adlen, + const unsigned char *npub, + const unsigned char *k) +{ + unsigned long long mlen = 0ULL; + int ret = -1; + + if (clen >= crypto_aead_xchacha20poly1305_ietf_ABYTES) { + ret = crypto_aead_xchacha20poly1305_ietf_decrypt_detached + (m, nsec, + c, clen - crypto_aead_xchacha20poly1305_ietf_ABYTES, + c + clen - crypto_aead_xchacha20poly1305_ietf_ABYTES, + ad, adlen, npub, k); + } + if (mlen_p != NULL) { + if (ret == 0) { + mlen = clen - crypto_aead_xchacha20poly1305_ietf_ABYTES; + } + *mlen_p = mlen; + } + return ret; +} + +size_t +crypto_aead_xchacha20poly1305_ietf_keybytes(void) +{ + return crypto_aead_xchacha20poly1305_ietf_KEYBYTES; +} + +size_t +crypto_aead_xchacha20poly1305_ietf_npubbytes(void) +{ + return crypto_aead_xchacha20poly1305_ietf_NPUBBYTES; +} + +size_t +crypto_aead_xchacha20poly1305_ietf_nsecbytes(void) +{ + return crypto_aead_xchacha20poly1305_ietf_NSECBYTES; +} + +size_t +crypto_aead_xchacha20poly1305_ietf_abytes(void) +{ + return crypto_aead_xchacha20poly1305_ietf_ABYTES; +} + +size_t +crypto_aead_xchacha20poly1305_ietf_messagebytes_max(void) +{ + return crypto_aead_xchacha20poly1305_ietf_MESSAGEBYTES_MAX; +} + +void +crypto_aead_xchacha20poly1305_ietf_keygen(unsigned char k[crypto_aead_xchacha20poly1305_ietf_KEYBYTES]) +{ + randombytes_buf(k, crypto_aead_xchacha20poly1305_ietf_KEYBYTES); +} diff --git a/sodium/sodium_blake2.h b/sodium/sodium_blake2.h new file mode 100644 index 00000000..ad46d82b --- /dev/null +++ b/sodium/sodium_blake2.h @@ -0,0 +1,102 @@ +/* + BLAKE2 reference source code package - reference C implementations + + Written in 2012 by Samuel Neves + + To the extent possible under law, the author(s) have dedicated all copyright + and related and neighboring rights to this software to the public domain + worldwide. This software is distributed without any warranty. + + All code is triple-licensed under the + [CC0](http://creativecommons.org/publicdomain/zero/1.0), the + [OpenSSL Licence](https://www.openssl.org/source/license.html), or + the [Apache Public License 2.0](http://www.apache.org/licenses/LICENSE-2.0), + at your choosing. + */ + +#ifndef blake2_H +#define blake2_H + +#include +#include + +#include "sodium_crypto_generichash_blake2b.h" +#include "sodium_export.h" + +enum blake2b_constant { + BLAKE2B_BLOCKBYTES = 128, + BLAKE2B_OUTBYTES = 64, + BLAKE2B_KEYBYTES = 64, + BLAKE2B_SALTBYTES = 16, + BLAKE2B_PERSONALBYTES = 16 +}; + +#if defined(__IBMC__) || defined(__SUNPRO_C) || defined(__SUNPRO_CC) +#pragma pack(1) +#else +#pragma pack(push, 1) +#endif + +typedef struct blake2b_param_ { + uint8_t digest_length; /* 1 */ + uint8_t key_length; /* 2 */ + uint8_t fanout; /* 3 */ + uint8_t depth; /* 4 */ + uint8_t leaf_length[4]; /* 8 */ + uint8_t node_offset[8]; /* 16 */ + uint8_t node_depth; /* 17 */ + uint8_t inner_length; /* 18 */ + uint8_t reserved[14]; /* 32 */ + uint8_t salt[BLAKE2B_SALTBYTES]; /* 48 */ + uint8_t personal[BLAKE2B_PERSONALBYTES]; /* 64 */ +} blake2b_param; + +typedef struct blake2b_state { + uint64_t h[8]; + uint64_t t[2]; + uint64_t f[2]; + uint8_t buf[2 * 128]; + size_t buflen; + uint8_t last_node; +} blake2b_state; + +#if defined(__IBMC__) || defined(__SUNPRO_C) || defined(__SUNPRO_CC) +#pragma pack() +#else +#pragma pack(pop) +#endif + +/* Streaming API */ +int blake2b_init(blake2b_state *S, const uint8_t outlen); +int blake2b_init_salt_personal(blake2b_state *S, const uint8_t outlen, + const void *salt, const void *personal); +int blake2b_init_key(blake2b_state *S, const uint8_t outlen, const void *key, + const uint8_t keylen); +int blake2b_init_key_salt_personal(blake2b_state *S, const uint8_t outlen, + const void *key, const uint8_t keylen, + const void *salt, const void *personal); +int blake2b_init_param(blake2b_state *S, const blake2b_param *P); +int blake2b_update(blake2b_state *S, const uint8_t *in, uint64_t inlen); +int blake2b_final(blake2b_state *S, uint8_t *out, uint8_t outlen); + +/* Simple API */ +int blake2b(uint8_t *out, const void *in, const void *key, const uint8_t outlen, + const uint64_t inlen, uint8_t keylen); +int blake2b_salt_personal(uint8_t *out, const void *in, const void *key, + const uint8_t outlen, const uint64_t inlen, + uint8_t keylen, const void *salt, + const void *personal); + +typedef int (*blake2b_compress_fn)(blake2b_state *S, + const uint8_t block[BLAKE2B_BLOCKBYTES]); +int blake2b_pick_best_implementation(void); +int blake2b_compress_ref(blake2b_state *S, + const uint8_t block[BLAKE2B_BLOCKBYTES]); +int blake2b_compress_ssse3(blake2b_state *S, + const uint8_t block[BLAKE2B_BLOCKBYTES]); +int blake2b_compress_sse41(blake2b_state *S, + const uint8_t block[BLAKE2B_BLOCKBYTES]); +int blake2b_compress_avx2(blake2b_state *S, + const uint8_t block[BLAKE2B_BLOCKBYTES]); + +#endif diff --git a/sodium/sodium_blake2b-compress-avx2.c b/sodium/sodium_blake2b-compress-avx2.c new file mode 100644 index 00000000..f9fd77d2 --- /dev/null +++ b/sodium/sodium_blake2b-compress-avx2.c @@ -0,0 +1,51 @@ + +#define BLAKE2_USE_SSSE3 +#define BLAKE2_USE_SSE41 +#define BLAKE2_USE_AVX2 + +#include +#include + +#include "sodium_blake2.h" +#include "sodium_private_common.h" +#include "sodium_private_sse2_64_32.h" + +#if defined(HAVE_AVX2INTRIN_H) && defined(HAVE_EMMINTRIN_H) && \ + defined(HAVE_TMMINTRIN_H) && defined(HAVE_SMMINTRIN_H) + +# ifdef __GNUC__ +# pragma GCC target("sse2") +# pragma GCC target("ssse3") +# pragma GCC target("sse4.1") +# pragma GCC target("avx2") +# endif + +# include +# include +# include +# include + +# include "sodium_blake2b-compress-avx2.h" + +CRYPTO_ALIGN(64) +static const uint64_t blake2b_IV[8] = { + 0x6a09e667f3bcc908ULL, 0xbb67ae8584caa73bULL, 0x3c6ef372fe94f82bULL, + 0xa54ff53a5f1d36f1ULL, 0x510e527fade682d1ULL, 0x9b05688c2b3e6c1fULL, + 0x1f83d9abfb41bd6bULL, 0x5be0cd19137e2179ULL +}; + +int +blake2b_compress_avx2(blake2b_state *S, const uint8_t block[BLAKE2B_BLOCKBYTES]) +{ + __m256i a = LOADU(&S->h[0]); + __m256i b = LOADU(&S->h[4]); + BLAKE2B_COMPRESS_V1(a, b, block, S->t[0], S->t[1], S->f[0], S->f[1]); + STOREU(&S->h[0], a); + STOREU(&S->h[4], b); + + return 0; +} + +#endif + +int blake2b_compress_avx2_link_warning_dummy = 0; diff --git a/sodium/sodium_blake2b-compress-avx2.h b/sodium/sodium_blake2b-compress-avx2.h new file mode 100644 index 00000000..a95211da --- /dev/null +++ b/sodium/sodium_blake2b-compress-avx2.h @@ -0,0 +1,142 @@ + +#ifndef blake2b_compress_avx2_H +#define blake2b_compress_avx2_H + +#define LOADU128(p) _mm_loadu_si128((const __m128i *) (p)) +#define STOREU128(p, r) _mm_storeu_si128((__m128i *) (p), r) + +#define LOADU(p) _mm256_loadu_si256((const __m256i *) (p)) +#define STOREU(p, r) _mm256_storeu_si256((__m256i *) (p), r) + +#if defined(__INTEL_COMPILER) || defined(_MSC_VER) || defined(__GNUC__) +# define LOAD(p) _mm256_load_si256((const __m256i *) (p)) +# define STORE(p, r) _mm256_store_si256((__m256i *) (p), r) +#else +# define LOAD(p) LOADU(p) +# define STORE(p, r) STOREU(p, r) +#endif + +static inline uint64_t +LOADU64(const void *p) +{ + uint64_t v; + memcpy(&v, p, sizeof v); + return v; +} + +#define ROTATE16 \ + _mm256_setr_epi8(2, 3, 4, 5, 6, 7, 0, 1, 10, 11, 12, 13, 14, 15, 8, 9, 2, \ + 3, 4, 5, 6, 7, 0, 1, 10, 11, 12, 13, 14, 15, 8, 9) + +#define ROTATE24 \ + _mm256_setr_epi8(3, 4, 5, 6, 7, 0, 1, 2, 11, 12, 13, 14, 15, 8, 9, 10, 3, \ + 4, 5, 6, 7, 0, 1, 2, 11, 12, 13, 14, 15, 8, 9, 10) + +#define ADD(a, b) _mm256_add_epi64(a, b) +#define SUB(a, b) _mm256_sub_epi64(a, b) + +#define XOR(a, b) _mm256_xor_si256(a, b) +#define AND(a, b) _mm256_and_si256(a, b) +#define OR(a, b) _mm256_or_si256(a, b) + +#define ROT32(x) _mm256_shuffle_epi32((x), _MM_SHUFFLE(2, 3, 0, 1)) +#define ROT24(x) _mm256_shuffle_epi8((x), ROTATE24) +#define ROT16(x) _mm256_shuffle_epi8((x), ROTATE16) +#define ROT63(x) _mm256_or_si256(_mm256_srli_epi64((x), 63), ADD((x), (x))) + +#define BLAKE2B_G1_V1(a, b, c, d, m) \ + do { \ + a = ADD(a, m); \ + a = ADD(a, b); \ + d = XOR(d, a); \ + d = ROT32(d); \ + c = ADD(c, d); \ + b = XOR(b, c); \ + b = ROT24(b); \ + } while (0) + +#define BLAKE2B_G2_V1(a, b, c, d, m) \ + do { \ + a = ADD(a, m); \ + a = ADD(a, b); \ + d = XOR(d, a); \ + d = ROT16(d); \ + c = ADD(c, d); \ + b = XOR(b, c); \ + b = ROT63(b); \ + } while (0) + +#define BLAKE2B_DIAG_V1(a, b, c, d) \ + do { \ + d = _mm256_permute4x64_epi64(d, _MM_SHUFFLE(2, 1, 0, 3)); \ + c = _mm256_permute4x64_epi64(c, _MM_SHUFFLE(1, 0, 3, 2)); \ + b = _mm256_permute4x64_epi64(b, _MM_SHUFFLE(0, 3, 2, 1)); \ + } while (0) + +#define BLAKE2B_UNDIAG_V1(a, b, c, d) \ + do { \ + d = _mm256_permute4x64_epi64(d, _MM_SHUFFLE(0, 3, 2, 1)); \ + c = _mm256_permute4x64_epi64(c, _MM_SHUFFLE(1, 0, 3, 2)); \ + b = _mm256_permute4x64_epi64(b, _MM_SHUFFLE(2, 1, 0, 3)); \ + } while (0) + +#include "sodium_blake2b-load-avx2.h" + +#define BLAKE2B_ROUND_V1(a, b, c, d, r, m) \ + do { \ + __m256i b0; \ + BLAKE2B_LOAD_MSG_##r##_1(b0); \ + BLAKE2B_G1_V1(a, b, c, d, b0); \ + BLAKE2B_LOAD_MSG_##r##_2(b0); \ + BLAKE2B_G2_V1(a, b, c, d, b0); \ + BLAKE2B_DIAG_V1(a, b, c, d); \ + BLAKE2B_LOAD_MSG_##r##_3(b0); \ + BLAKE2B_G1_V1(a, b, c, d, b0); \ + BLAKE2B_LOAD_MSG_##r##_4(b0); \ + BLAKE2B_G2_V1(a, b, c, d, b0); \ + BLAKE2B_UNDIAG_V1(a, b, c, d); \ + } while (0) + +#define BLAKE2B_ROUNDS_V1(a, b, c, d, m) \ + do { \ + BLAKE2B_ROUND_V1(a, b, c, d, 0, (m)); \ + BLAKE2B_ROUND_V1(a, b, c, d, 1, (m)); \ + BLAKE2B_ROUND_V1(a, b, c, d, 2, (m)); \ + BLAKE2B_ROUND_V1(a, b, c, d, 3, (m)); \ + BLAKE2B_ROUND_V1(a, b, c, d, 4, (m)); \ + BLAKE2B_ROUND_V1(a, b, c, d, 5, (m)); \ + BLAKE2B_ROUND_V1(a, b, c, d, 6, (m)); \ + BLAKE2B_ROUND_V1(a, b, c, d, 7, (m)); \ + BLAKE2B_ROUND_V1(a, b, c, d, 8, (m)); \ + BLAKE2B_ROUND_V1(a, b, c, d, 9, (m)); \ + BLAKE2B_ROUND_V1(a, b, c, d, 10, (m)); \ + BLAKE2B_ROUND_V1(a, b, c, d, 11, (m)); \ + } while (0) + +#define DECLARE_MESSAGE_WORDS(m) \ + const __m256i m0 = _mm256_broadcastsi128_si256(LOADU128((m) + 0)); \ + const __m256i m1 = _mm256_broadcastsi128_si256(LOADU128((m) + 16)); \ + const __m256i m2 = _mm256_broadcastsi128_si256(LOADU128((m) + 32)); \ + const __m256i m3 = _mm256_broadcastsi128_si256(LOADU128((m) + 48)); \ + const __m256i m4 = _mm256_broadcastsi128_si256(LOADU128((m) + 64)); \ + const __m256i m5 = _mm256_broadcastsi128_si256(LOADU128((m) + 80)); \ + const __m256i m6 = _mm256_broadcastsi128_si256(LOADU128((m) + 96)); \ + const __m256i m7 = _mm256_broadcastsi128_si256(LOADU128((m) + 112)); \ + __m256i t0, t1; + +#define BLAKE2B_COMPRESS_V1(a, b, m, t0, t1, f0, f1) \ + do { \ + DECLARE_MESSAGE_WORDS(m) \ + const __m256i iv0 = a; \ + const __m256i iv1 = b; \ + __m256i c = LOAD(&blake2b_IV[0]); \ + __m256i d = \ + XOR(LOAD(&blake2b_IV[4]), _mm256_set_epi64x(f1, f0, t1, t0)); \ + BLAKE2B_ROUNDS_V1(a, b, c, d, m); \ + a = XOR(a, c); \ + b = XOR(b, d); \ + a = XOR(a, iv0); \ + b = XOR(b, iv1); \ + } while (0) + +#endif diff --git a/sodium/sodium_blake2b-compress-ref.c b/sodium/sodium_blake2b-compress-ref.c new file mode 100644 index 00000000..3f861511 --- /dev/null +++ b/sodium/sodium_blake2b-compress-ref.c @@ -0,0 +1,93 @@ + +#include +#include + +#include "sodium_blake2.h" +#include "sodium_private_common.h" + +CRYPTO_ALIGN(64) +static const uint64_t blake2b_IV[8] = { + 0x6a09e667f3bcc908ULL, 0xbb67ae8584caa73bULL, 0x3c6ef372fe94f82bULL, + 0xa54ff53a5f1d36f1ULL, 0x510e527fade682d1ULL, 0x9b05688c2b3e6c1fULL, + 0x1f83d9abfb41bd6bULL, 0x5be0cd19137e2179ULL +}; + +static const uint8_t blake2b_sigma[12][16] = { + { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 }, + { 14, 10, 4, 8, 9, 15, 13, 6, 1, 12, 0, 2, 11, 7, 5, 3 }, + { 11, 8, 12, 0, 5, 2, 15, 13, 10, 14, 3, 6, 7, 1, 9, 4 }, + { 7, 9, 3, 1, 13, 12, 11, 14, 2, 6, 5, 10, 4, 0, 15, 8 }, + { 9, 0, 5, 7, 2, 4, 10, 15, 14, 1, 11, 12, 6, 8, 3, 13 }, + { 2, 12, 6, 10, 0, 11, 8, 3, 4, 13, 7, 5, 15, 14, 1, 9 }, + { 12, 5, 1, 15, 14, 13, 4, 10, 0, 7, 6, 3, 9, 2, 8, 11 }, + { 13, 11, 7, 14, 12, 1, 3, 9, 5, 0, 15, 4, 8, 6, 2, 10 }, + { 6, 15, 14, 9, 11, 3, 0, 8, 12, 2, 13, 7, 1, 4, 10, 5 }, + { 10, 2, 8, 4, 7, 6, 1, 5, 15, 11, 9, 14, 3, 12, 13, 0 }, + { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 }, + { 14, 10, 4, 8, 9, 15, 13, 6, 1, 12, 0, 2, 11, 7, 5, 3 } +}; + +int +blake2b_compress_ref(blake2b_state *S, const uint8_t block[BLAKE2B_BLOCKBYTES]) +{ + uint64_t m[16]; + uint64_t v[16]; + int i; + + for (i = 0; i < 16; ++i) { + m[i] = LOAD64_LE(block + i * sizeof(m[i])); + } + for (i = 0; i < 8; ++i) { + v[i] = S->h[i]; + } + v[8] = blake2b_IV[0]; + v[9] = blake2b_IV[1]; + v[10] = blake2b_IV[2]; + v[11] = blake2b_IV[3]; + v[12] = S->t[0] ^ blake2b_IV[4]; + v[13] = S->t[1] ^ blake2b_IV[5]; + v[14] = S->f[0] ^ blake2b_IV[6]; + v[15] = S->f[1] ^ blake2b_IV[7]; +#define G(r, i, a, b, c, d) \ + do { \ + a = a + b + m[blake2b_sigma[r][2 * i + 0]]; \ + d = ROTR64(d ^ a, 32); \ + c = c + d; \ + b = ROTR64(b ^ c, 24); \ + a = a + b + m[blake2b_sigma[r][2 * i + 1]]; \ + d = ROTR64(d ^ a, 16); \ + c = c + d; \ + b = ROTR64(b ^ c, 63); \ + } while (0) +#define ROUND(r) \ + do { \ + G(r, 0, v[0], v[4], v[8], v[12]); \ + G(r, 1, v[1], v[5], v[9], v[13]); \ + G(r, 2, v[2], v[6], v[10], v[14]); \ + G(r, 3, v[3], v[7], v[11], v[15]); \ + G(r, 4, v[0], v[5], v[10], v[15]); \ + G(r, 5, v[1], v[6], v[11], v[12]); \ + G(r, 6, v[2], v[7], v[8], v[13]); \ + G(r, 7, v[3], v[4], v[9], v[14]); \ + } while (0) + ROUND(0); + ROUND(1); + ROUND(2); + ROUND(3); + ROUND(4); + ROUND(5); + ROUND(6); + ROUND(7); + ROUND(8); + ROUND(9); + ROUND(10); + ROUND(11); + + for (i = 0; i < 8; ++i) { + S->h[i] = S->h[i] ^ v[i] ^ v[i + 8]; + } + +#undef G +#undef ROUND + return 0; +} diff --git a/sodium/sodium_blake2b-compress-sse41.c b/sodium/sodium_blake2b-compress-sse41.c new file mode 100644 index 00000000..e3f9e36a --- /dev/null +++ b/sodium/sodium_blake2b-compress-sse41.c @@ -0,0 +1,89 @@ + +#define BLAKE2_USE_SSSE3 +#define BLAKE2_USE_SSE41 + +#include +#include + +#include "sodium_blake2.h" +#include "sodium_private_common.h" +#include "sodium_private_sse2_64_32.h" + +#if defined(HAVE_EMMINTRIN_H) && defined(HAVE_TMMINTRIN_H) && \ + defined(HAVE_SMMINTRIN_H) + +# ifdef __GNUC__ +# pragma GCC target("sse2") +# pragma GCC target("ssse3") +# pragma GCC target("sse4.1") +# endif + +# include +# include +# include + +# include "sodium_blake2b-compress-sse41.h" + +CRYPTO_ALIGN(64) +static const uint64_t blake2b_IV[8] = { + 0x6a09e667f3bcc908ULL, 0xbb67ae8584caa73bULL, 0x3c6ef372fe94f82bULL, + 0xa54ff53a5f1d36f1ULL, 0x510e527fade682d1ULL, 0x9b05688c2b3e6c1fULL, + 0x1f83d9abfb41bd6bULL, 0x5be0cd19137e2179ULL +}; + +int +blake2b_compress_sse41(blake2b_state *S, + const uint8_t block[BLAKE2B_BLOCKBYTES]) +{ + __m128i row1l, row1h; + __m128i row2l, row2h; + __m128i row3l, row3h; + __m128i row4l, row4h; + __m128i b0, b1; + __m128i t0, t1; + const __m128i r16 = + _mm_setr_epi8(2, 3, 4, 5, 6, 7, 0, 1, 10, 11, 12, 13, 14, 15, 8, 9); + const __m128i r24 = + _mm_setr_epi8(3, 4, 5, 6, 7, 0, 1, 2, 11, 12, 13, 14, 15, 8, 9, 10); + const __m128i m0 = LOADU(block + 00); + const __m128i m1 = LOADU(block + 16); + const __m128i m2 = LOADU(block + 32); + const __m128i m3 = LOADU(block + 48); + const __m128i m4 = LOADU(block + 64); + const __m128i m5 = LOADU(block + 80); + const __m128i m6 = LOADU(block + 96); + const __m128i m7 = LOADU(block + 112); + row1l = LOADU(&S->h[0]); + row1h = LOADU(&S->h[2]); + row2l = LOADU(&S->h[4]); + row2h = LOADU(&S->h[6]); + row3l = LOADU(&blake2b_IV[0]); + row3h = LOADU(&blake2b_IV[2]); + row4l = _mm_xor_si128(LOADU(&blake2b_IV[4]), LOADU(&S->t[0])); + row4h = _mm_xor_si128(LOADU(&blake2b_IV[6]), LOADU(&S->f[0])); + ROUND(0); + ROUND(1); + ROUND(2); + ROUND(3); + ROUND(4); + ROUND(5); + ROUND(6); + ROUND(7); + ROUND(8); + ROUND(9); + ROUND(10); + ROUND(11); + row1l = _mm_xor_si128(row3l, row1l); + row1h = _mm_xor_si128(row3h, row1h); + STOREU(&S->h[0], _mm_xor_si128(LOADU(&S->h[0]), row1l)); + STOREU(&S->h[2], _mm_xor_si128(LOADU(&S->h[2]), row1h)); + row2l = _mm_xor_si128(row4l, row2l); + row2h = _mm_xor_si128(row4h, row2h); + STOREU(&S->h[4], _mm_xor_si128(LOADU(&S->h[4]), row2l)); + STOREU(&S->h[6], _mm_xor_si128(LOADU(&S->h[6]), row2h)); + return 0; +} + +#endif + +int blake2b_compress_sse41_link_warning_dummy = 0; diff --git a/sodium/sodium_blake2b-compress-sse41.h b/sodium/sodium_blake2b-compress-sse41.h new file mode 100644 index 00000000..b73362cb --- /dev/null +++ b/sodium/sodium_blake2b-compress-sse41.h @@ -0,0 +1,103 @@ + +#ifndef blake2b_compress_sse41_H +#define blake2b_compress_sse41_H + +#define LOADU(p) _mm_loadu_si128((const __m128i *) (const void *) (p)) +#define STOREU(p, r) _mm_storeu_si128((__m128i *) (void *) (p), r) + +#define _mm_roti_epi64(x, c) \ + (-(c) == 32) \ + ? _mm_shuffle_epi32((x), _MM_SHUFFLE(2, 3, 0, 1)) \ + : (-(c) == 24) \ + ? _mm_shuffle_epi8((x), r24) \ + : (-(c) == 16) \ + ? _mm_shuffle_epi8((x), r16) \ + : (-(c) == 63) \ + ? _mm_xor_si128(_mm_srli_epi64((x), -(c)), \ + _mm_add_epi64((x), (x))) \ + : _mm_xor_si128(_mm_srli_epi64((x), -(c)), \ + _mm_slli_epi64((x), 64 - (-(c)))) + +#define G1(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1) \ + row1l = _mm_add_epi64(_mm_add_epi64(row1l, b0), row2l); \ + row1h = _mm_add_epi64(_mm_add_epi64(row1h, b1), row2h); \ + \ + row4l = _mm_xor_si128(row4l, row1l); \ + row4h = _mm_xor_si128(row4h, row1h); \ + \ + row4l = _mm_roti_epi64(row4l, -32); \ + row4h = _mm_roti_epi64(row4h, -32); \ + \ + row3l = _mm_add_epi64(row3l, row4l); \ + row3h = _mm_add_epi64(row3h, row4h); \ + \ + row2l = _mm_xor_si128(row2l, row3l); \ + row2h = _mm_xor_si128(row2h, row3h); \ + \ + row2l = _mm_roti_epi64(row2l, -24); \ + row2h = _mm_roti_epi64(row2h, -24); + +#define G2(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1) \ + row1l = _mm_add_epi64(_mm_add_epi64(row1l, b0), row2l); \ + row1h = _mm_add_epi64(_mm_add_epi64(row1h, b1), row2h); \ + \ + row4l = _mm_xor_si128(row4l, row1l); \ + row4h = _mm_xor_si128(row4h, row1h); \ + \ + row4l = _mm_roti_epi64(row4l, -16); \ + row4h = _mm_roti_epi64(row4h, -16); \ + \ + row3l = _mm_add_epi64(row3l, row4l); \ + row3h = _mm_add_epi64(row3h, row4h); \ + \ + row2l = _mm_xor_si128(row2l, row3l); \ + row2h = _mm_xor_si128(row2h, row3h); \ + \ + row2l = _mm_roti_epi64(row2l, -63); \ + row2h = _mm_roti_epi64(row2h, -63); + +#define DIAGONALIZE(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h) \ + t0 = _mm_alignr_epi8(row2h, row2l, 8); \ + t1 = _mm_alignr_epi8(row2l, row2h, 8); \ + row2l = t0; \ + row2h = t1; \ + \ + t0 = row3l; \ + row3l = row3h; \ + row3h = t0; \ + \ + t0 = _mm_alignr_epi8(row4h, row4l, 8); \ + t1 = _mm_alignr_epi8(row4l, row4h, 8); \ + row4l = t1; \ + row4h = t0; + +#define UNDIAGONALIZE(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h) \ + t0 = _mm_alignr_epi8(row2l, row2h, 8); \ + t1 = _mm_alignr_epi8(row2h, row2l, 8); \ + row2l = t0; \ + row2h = t1; \ + \ + t0 = row3l; \ + row3l = row3h; \ + row3h = t0; \ + \ + t0 = _mm_alignr_epi8(row4l, row4h, 8); \ + t1 = _mm_alignr_epi8(row4h, row4l, 8); \ + row4l = t1; \ + row4h = t0; + +#include "sodium_blake2b-load-sse41.h" + +#define ROUND(r) \ + LOAD_MSG_##r##_1(b0, b1); \ + G1(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1); \ + LOAD_MSG_##r##_2(b0, b1); \ + G2(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1); \ + DIAGONALIZE(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h); \ + LOAD_MSG_##r##_3(b0, b1); \ + G1(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1); \ + LOAD_MSG_##r##_4(b0, b1); \ + G2(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1); \ + UNDIAGONALIZE(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h); + +#endif diff --git a/sodium/sodium_blake2b-compress-ssse3.c b/sodium/sodium_blake2b-compress-ssse3.c new file mode 100644 index 00000000..c87a959c --- /dev/null +++ b/sodium/sodium_blake2b-compress-ssse3.c @@ -0,0 +1,92 @@ + +#include +#include + +#include "sodium_blake2.h" +#include "sodium_private_common.h" +#include "sodium_private_sse2_64_32.h" + +#if defined(HAVE_EMMINTRIN_H) && defined(HAVE_TMMINTRIN_H) + +# ifdef __GNUC__ +# pragma GCC target("sse2") +# pragma GCC target("ssse3") +# endif + +# include +# include + +# include "sodium_blake2b-compress-ssse3.h" + +CRYPTO_ALIGN(64) +static const uint64_t blake2b_IV[8] = { + 0x6a09e667f3bcc908ULL, 0xbb67ae8584caa73bULL, 0x3c6ef372fe94f82bULL, + 0xa54ff53a5f1d36f1ULL, 0x510e527fade682d1ULL, 0x9b05688c2b3e6c1fULL, + 0x1f83d9abfb41bd6bULL, 0x5be0cd19137e2179ULL +}; + +int +blake2b_compress_ssse3(blake2b_state *S, + const uint8_t block[BLAKE2B_BLOCKBYTES]) +{ + __m128i row1l, row1h; + __m128i row2l, row2h; + __m128i row3l, row3h; + __m128i row4l, row4h; + __m128i b0, b1; + __m128i t0, t1; + const __m128i r16 = + _mm_setr_epi8(2, 3, 4, 5, 6, 7, 0, 1, 10, 11, 12, 13, 14, 15, 8, 9); + const __m128i r24 = + _mm_setr_epi8(3, 4, 5, 6, 7, 0, 1, 2, 11, 12, 13, 14, 15, 8, 9, 10); + const uint64_t m0 = ((const uint64_t *) block)[0]; + const uint64_t m1 = ((const uint64_t *) block)[1]; + const uint64_t m2 = ((const uint64_t *) block)[2]; + const uint64_t m3 = ((const uint64_t *) block)[3]; + const uint64_t m4 = ((const uint64_t *) block)[4]; + const uint64_t m5 = ((const uint64_t *) block)[5]; + const uint64_t m6 = ((const uint64_t *) block)[6]; + const uint64_t m7 = ((const uint64_t *) block)[7]; + const uint64_t m8 = ((const uint64_t *) block)[8]; + const uint64_t m9 = ((const uint64_t *) block)[9]; + const uint64_t m10 = ((const uint64_t *) block)[10]; + const uint64_t m11 = ((const uint64_t *) block)[11]; + const uint64_t m12 = ((const uint64_t *) block)[12]; + const uint64_t m13 = ((const uint64_t *) block)[13]; + const uint64_t m14 = ((const uint64_t *) block)[14]; + const uint64_t m15 = ((const uint64_t *) block)[15]; + + row1l = LOADU(&S->h[0]); + row1h = LOADU(&S->h[2]); + row2l = LOADU(&S->h[4]); + row2h = LOADU(&S->h[6]); + row3l = LOADU(&blake2b_IV[0]); + row3h = LOADU(&blake2b_IV[2]); + row4l = _mm_xor_si128(LOADU(&blake2b_IV[4]), LOADU(&S->t[0])); + row4h = _mm_xor_si128(LOADU(&blake2b_IV[6]), LOADU(&S->f[0])); + ROUND(0); + ROUND(1); + ROUND(2); + ROUND(3); + ROUND(4); + ROUND(5); + ROUND(6); + ROUND(7); + ROUND(8); + ROUND(9); + ROUND(10); + ROUND(11); + row1l = _mm_xor_si128(row3l, row1l); + row1h = _mm_xor_si128(row3h, row1h); + STOREU(&S->h[0], _mm_xor_si128(LOADU(&S->h[0]), row1l)); + STOREU(&S->h[2], _mm_xor_si128(LOADU(&S->h[2]), row1h)); + row2l = _mm_xor_si128(row4l, row2l); + row2h = _mm_xor_si128(row4h, row2h); + STOREU(&S->h[4], _mm_xor_si128(LOADU(&S->h[4]), row2l)); + STOREU(&S->h[6], _mm_xor_si128(LOADU(&S->h[6]), row2h)); + return 0; +} + +#endif + +int blake2b_compress_sse3_link_warning_dummy = 0; diff --git a/sodium/sodium_blake2b-compress-ssse3.h b/sodium/sodium_blake2b-compress-ssse3.h new file mode 100644 index 00000000..751fef2e --- /dev/null +++ b/sodium/sodium_blake2b-compress-ssse3.h @@ -0,0 +1,103 @@ + +#ifndef blake2b_compress_ssse3_H +#define blake2b_compress_ssse3_H + +#define LOADU(p) _mm_loadu_si128((const __m128i *) (const void *) (p)) +#define STOREU(p, r) _mm_storeu_si128((__m128i *) (void *) (p), r) + +#define _mm_roti_epi64(x, c) \ + (-(c) == 32) \ + ? _mm_shuffle_epi32((x), _MM_SHUFFLE(2, 3, 0, 1)) \ + : (-(c) == 24) \ + ? _mm_shuffle_epi8((x), r24) \ + : (-(c) == 16) \ + ? _mm_shuffle_epi8((x), r16) \ + : (-(c) == 63) \ + ? _mm_xor_si128(_mm_srli_epi64((x), -(c)), \ + _mm_add_epi64((x), (x))) \ + : _mm_xor_si128(_mm_srli_epi64((x), -(c)), \ + _mm_slli_epi64((x), 64 - (-(c)))) + +#define G1(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1) \ + row1l = _mm_add_epi64(_mm_add_epi64(row1l, b0), row2l); \ + row1h = _mm_add_epi64(_mm_add_epi64(row1h, b1), row2h); \ + \ + row4l = _mm_xor_si128(row4l, row1l); \ + row4h = _mm_xor_si128(row4h, row1h); \ + \ + row4l = _mm_roti_epi64(row4l, -32); \ + row4h = _mm_roti_epi64(row4h, -32); \ + \ + row3l = _mm_add_epi64(row3l, row4l); \ + row3h = _mm_add_epi64(row3h, row4h); \ + \ + row2l = _mm_xor_si128(row2l, row3l); \ + row2h = _mm_xor_si128(row2h, row3h); \ + \ + row2l = _mm_roti_epi64(row2l, -24); \ + row2h = _mm_roti_epi64(row2h, -24); + +#define G2(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1) \ + row1l = _mm_add_epi64(_mm_add_epi64(row1l, b0), row2l); \ + row1h = _mm_add_epi64(_mm_add_epi64(row1h, b1), row2h); \ + \ + row4l = _mm_xor_si128(row4l, row1l); \ + row4h = _mm_xor_si128(row4h, row1h); \ + \ + row4l = _mm_roti_epi64(row4l, -16); \ + row4h = _mm_roti_epi64(row4h, -16); \ + \ + row3l = _mm_add_epi64(row3l, row4l); \ + row3h = _mm_add_epi64(row3h, row4h); \ + \ + row2l = _mm_xor_si128(row2l, row3l); \ + row2h = _mm_xor_si128(row2h, row3h); \ + \ + row2l = _mm_roti_epi64(row2l, -63); \ + row2h = _mm_roti_epi64(row2h, -63); + +#define DIAGONALIZE(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h) \ + t0 = _mm_alignr_epi8(row2h, row2l, 8); \ + t1 = _mm_alignr_epi8(row2l, row2h, 8); \ + row2l = t0; \ + row2h = t1; \ + \ + t0 = row3l; \ + row3l = row3h; \ + row3h = t0; \ + \ + t0 = _mm_alignr_epi8(row4h, row4l, 8); \ + t1 = _mm_alignr_epi8(row4l, row4h, 8); \ + row4l = t1; \ + row4h = t0; + +#define UNDIAGONALIZE(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h) \ + t0 = _mm_alignr_epi8(row2l, row2h, 8); \ + t1 = _mm_alignr_epi8(row2h, row2l, 8); \ + row2l = t0; \ + row2h = t1; \ + \ + t0 = row3l; \ + row3l = row3h; \ + row3h = t0; \ + \ + t0 = _mm_alignr_epi8(row4l, row4h, 8); \ + t1 = _mm_alignr_epi8(row4h, row4l, 8); \ + row4l = t1; \ + row4h = t0; + +#include "sodium_blake2b-load-sse2.h" + +#define ROUND(r) \ + LOAD_MSG_##r##_1(b0, b1); \ + G1(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1); \ + LOAD_MSG_##r##_2(b0, b1); \ + G2(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1); \ + DIAGONALIZE(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h); \ + LOAD_MSG_##r##_3(b0, b1); \ + G1(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1); \ + LOAD_MSG_##r##_4(b0, b1); \ + G2(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h, b0, b1); \ + UNDIAGONALIZE(row1l, row2l, row3l, row4l, row1h, row2h, row3h, row4h); + +#endif diff --git a/sodium/sodium_blake2b-load-avx2.h b/sodium/sodium_blake2b-load-avx2.h new file mode 100644 index 00000000..8c15f177 --- /dev/null +++ b/sodium/sodium_blake2b-load-avx2.h @@ -0,0 +1,340 @@ +#ifndef blake2b_load_avx2_H +#define blake2b_load_avx2_H + +#define BLAKE2B_LOAD_MSG_0_1(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m0, m1); \ + t1 = _mm256_unpacklo_epi64(m2, m3); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_0_2(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m0, m1); \ + t1 = _mm256_unpackhi_epi64(m2, m3); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_0_3(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m4, m5); \ + t1 = _mm256_unpacklo_epi64(m6, m7); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_0_4(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m4, m5); \ + t1 = _mm256_unpackhi_epi64(m6, m7); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_1_1(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m7, m2); \ + t1 = _mm256_unpackhi_epi64(m4, m6); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_1_2(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m5, m4); \ + t1 = _mm256_alignr_epi8(m3, m7, 8); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_1_3(b0) \ + do { \ + t0 = _mm256_shuffle_epi32(m0, _MM_SHUFFLE(1, 0, 3, 2)); \ + t1 = _mm256_unpackhi_epi64(m5, m2); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_1_4(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m6, m1); \ + t1 = _mm256_unpackhi_epi64(m3, m1); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_2_1(b0) \ + do { \ + t0 = _mm256_alignr_epi8(m6, m5, 8); \ + t1 = _mm256_unpackhi_epi64(m2, m7); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_2_2(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m4, m0); \ + t1 = _mm256_blend_epi32(m6, m1, 0x33); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_2_3(b0) \ + do { \ + t0 = _mm256_blend_epi32(m1, m5, 0x33); \ + t1 = _mm256_unpackhi_epi64(m3, m4); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_2_4(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m7, m3); \ + t1 = _mm256_alignr_epi8(m2, m0, 8); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_3_1(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m3, m1); \ + t1 = _mm256_unpackhi_epi64(m6, m5); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_3_2(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m4, m0); \ + t1 = _mm256_unpacklo_epi64(m6, m7); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_3_3(b0) \ + do { \ + t0 = _mm256_blend_epi32(m2, m1, 0x33); \ + t1 = _mm256_blend_epi32(m7, m2, 0x33); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_3_4(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m3, m5); \ + t1 = _mm256_unpacklo_epi64(m0, m4); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_4_1(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m4, m2); \ + t1 = _mm256_unpacklo_epi64(m1, m5); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_4_2(b0) \ + do { \ + t0 = _mm256_blend_epi32(m3, m0, 0x33); \ + t1 = _mm256_blend_epi32(m7, m2, 0x33); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_4_3(b0) \ + do { \ + t0 = _mm256_blend_epi32(m5, m7, 0x33); \ + t1 = _mm256_blend_epi32(m1, m3, 0x33); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_4_4(b0) \ + do { \ + t0 = _mm256_alignr_epi8(m6, m0, 8); \ + t1 = _mm256_blend_epi32(m6, m4, 0x33); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_5_1(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m1, m3); \ + t1 = _mm256_unpacklo_epi64(m0, m4); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_5_2(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m6, m5); \ + t1 = _mm256_unpackhi_epi64(m5, m1); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_5_3(b0) \ + do { \ + t0 = _mm256_blend_epi32(m3, m2, 0x33); \ + t1 = _mm256_unpackhi_epi64(m7, m0); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_5_4(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m6, m2); \ + t1 = _mm256_blend_epi32(m4, m7, 0x33); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_6_1(b0) \ + do { \ + t0 = _mm256_blend_epi32(m0, m6, 0x33); \ + t1 = _mm256_unpacklo_epi64(m7, m2); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_6_2(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m2, m7); \ + t1 = _mm256_alignr_epi8(m5, m6, 8); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_6_3(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m0, m3); \ + t1 = _mm256_shuffle_epi32(m4, _MM_SHUFFLE(1, 0, 3, 2)); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_6_4(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m3, m1); \ + t1 = _mm256_blend_epi32(m5, m1, 0x33); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_7_1(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m6, m3); \ + t1 = _mm256_blend_epi32(m1, m6, 0x33); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_7_2(b0) \ + do { \ + t0 = _mm256_alignr_epi8(m7, m5, 8); \ + t1 = _mm256_unpackhi_epi64(m0, m4); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_7_3(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m2, m7); \ + t1 = _mm256_unpacklo_epi64(m4, m1); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_7_4(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m0, m2); \ + t1 = _mm256_unpacklo_epi64(m3, m5); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_8_1(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m3, m7); \ + t1 = _mm256_alignr_epi8(m0, m5, 8); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_8_2(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m7, m4); \ + t1 = _mm256_alignr_epi8(m4, m1, 8); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_8_3(b0) \ + do { \ + t0 = m6; \ + t1 = _mm256_alignr_epi8(m5, m0, 8); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_8_4(b0) \ + do { \ + t0 = _mm256_blend_epi32(m3, m1, 0x33); \ + t1 = m2; \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_9_1(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m5, m4); \ + t1 = _mm256_unpackhi_epi64(m3, m0); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_9_2(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m1, m2); \ + t1 = _mm256_blend_epi32(m2, m3, 0x33); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_9_3(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m7, m4); \ + t1 = _mm256_unpackhi_epi64(m1, m6); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_9_4(b0) \ + do { \ + t0 = _mm256_alignr_epi8(m7, m5, 8); \ + t1 = _mm256_unpacklo_epi64(m6, m0); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_10_1(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m0, m1); \ + t1 = _mm256_unpacklo_epi64(m2, m3); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_10_2(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m0, m1); \ + t1 = _mm256_unpackhi_epi64(m2, m3); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_10_3(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m4, m5); \ + t1 = _mm256_unpacklo_epi64(m6, m7); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_10_4(b0) \ + do { \ + t0 = _mm256_unpackhi_epi64(m4, m5); \ + t1 = _mm256_unpackhi_epi64(m6, m7); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_11_1(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m7, m2); \ + t1 = _mm256_unpackhi_epi64(m4, m6); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_11_2(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m5, m4); \ + t1 = _mm256_alignr_epi8(m3, m7, 8); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_11_3(b0) \ + do { \ + t0 = _mm256_shuffle_epi32(m0, _MM_SHUFFLE(1, 0, 3, 2)); \ + t1 = _mm256_unpackhi_epi64(m5, m2); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#define BLAKE2B_LOAD_MSG_11_4(b0) \ + do { \ + t0 = _mm256_unpacklo_epi64(m6, m1); \ + t1 = _mm256_unpackhi_epi64(m3, m1); \ + b0 = _mm256_blend_epi32(t0, t1, 0xF0); \ + } while (0) + +#endif diff --git a/sodium/sodium_blake2b-load-sse2.h b/sodium/sodium_blake2b-load-sse2.h new file mode 100644 index 00000000..8e67421a --- /dev/null +++ b/sodium/sodium_blake2b-load-sse2.h @@ -0,0 +1,164 @@ +/* + BLAKE2 reference source code package - optimized C implementations + + Written in 2012 by Samuel Neves + + To the extent possible under law, the author(s) have dedicated all copyright + and related and neighboring rights to this software to the public domain + worldwide. This software is distributed without any warranty. + + You should have received a copy of the CC0 Public Domain Dedication along + with + this software. If not, see + . +*/ + +#ifndef blake2b_load_sse2_H +#define blake2b_load_sse2_H + +#define LOAD_MSG_0_1(b0, b1) \ + b0 = _mm_set_epi64x(m2, m0); \ + b1 = _mm_set_epi64x(m6, m4) +#define LOAD_MSG_0_2(b0, b1) \ + b0 = _mm_set_epi64x(m3, m1); \ + b1 = _mm_set_epi64x(m7, m5) +#define LOAD_MSG_0_3(b0, b1) \ + b0 = _mm_set_epi64x(m10, m8); \ + b1 = _mm_set_epi64x(m14, m12) +#define LOAD_MSG_0_4(b0, b1) \ + b0 = _mm_set_epi64x(m11, m9); \ + b1 = _mm_set_epi64x(m15, m13) +#define LOAD_MSG_1_1(b0, b1) \ + b0 = _mm_set_epi64x(m4, m14); \ + b1 = _mm_set_epi64x(m13, m9) +#define LOAD_MSG_1_2(b0, b1) \ + b0 = _mm_set_epi64x(m8, m10); \ + b1 = _mm_set_epi64x(m6, m15) +#define LOAD_MSG_1_3(b0, b1) \ + b0 = _mm_set_epi64x(m0, m1); \ + b1 = _mm_set_epi64x(m5, m11) +#define LOAD_MSG_1_4(b0, b1) \ + b0 = _mm_set_epi64x(m2, m12); \ + b1 = _mm_set_epi64x(m3, m7) +#define LOAD_MSG_2_1(b0, b1) \ + b0 = _mm_set_epi64x(m12, m11); \ + b1 = _mm_set_epi64x(m15, m5) +#define LOAD_MSG_2_2(b0, b1) \ + b0 = _mm_set_epi64x(m0, m8); \ + b1 = _mm_set_epi64x(m13, m2) +#define LOAD_MSG_2_3(b0, b1) \ + b0 = _mm_set_epi64x(m3, m10); \ + b1 = _mm_set_epi64x(m9, m7) +#define LOAD_MSG_2_4(b0, b1) \ + b0 = _mm_set_epi64x(m6, m14); \ + b1 = _mm_set_epi64x(m4, m1) +#define LOAD_MSG_3_1(b0, b1) \ + b0 = _mm_set_epi64x(m3, m7); \ + b1 = _mm_set_epi64x(m11, m13) +#define LOAD_MSG_3_2(b0, b1) \ + b0 = _mm_set_epi64x(m1, m9); \ + b1 = _mm_set_epi64x(m14, m12) +#define LOAD_MSG_3_3(b0, b1) \ + b0 = _mm_set_epi64x(m5, m2); \ + b1 = _mm_set_epi64x(m15, m4) +#define LOAD_MSG_3_4(b0, b1) \ + b0 = _mm_set_epi64x(m10, m6); \ + b1 = _mm_set_epi64x(m8, m0) +#define LOAD_MSG_4_1(b0, b1) \ + b0 = _mm_set_epi64x(m5, m9); \ + b1 = _mm_set_epi64x(m10, m2) +#define LOAD_MSG_4_2(b0, b1) \ + b0 = _mm_set_epi64x(m7, m0); \ + b1 = _mm_set_epi64x(m15, m4) +#define LOAD_MSG_4_3(b0, b1) \ + b0 = _mm_set_epi64x(m11, m14); \ + b1 = _mm_set_epi64x(m3, m6) +#define LOAD_MSG_4_4(b0, b1) \ + b0 = _mm_set_epi64x(m12, m1); \ + b1 = _mm_set_epi64x(m13, m8) +#define LOAD_MSG_5_1(b0, b1) \ + b0 = _mm_set_epi64x(m6, m2); \ + b1 = _mm_set_epi64x(m8, m0) +#define LOAD_MSG_5_2(b0, b1) \ + b0 = _mm_set_epi64x(m10, m12); \ + b1 = _mm_set_epi64x(m3, m11) +#define LOAD_MSG_5_3(b0, b1) \ + b0 = _mm_set_epi64x(m7, m4); \ + b1 = _mm_set_epi64x(m1, m15) +#define LOAD_MSG_5_4(b0, b1) \ + b0 = _mm_set_epi64x(m5, m13); \ + b1 = _mm_set_epi64x(m9, m14) +#define LOAD_MSG_6_1(b0, b1) \ + b0 = _mm_set_epi64x(m1, m12); \ + b1 = _mm_set_epi64x(m4, m14) +#define LOAD_MSG_6_2(b0, b1) \ + b0 = _mm_set_epi64x(m15, m5); \ + b1 = _mm_set_epi64x(m10, m13) +#define LOAD_MSG_6_3(b0, b1) \ + b0 = _mm_set_epi64x(m6, m0); \ + b1 = _mm_set_epi64x(m8, m9) +#define LOAD_MSG_6_4(b0, b1) \ + b0 = _mm_set_epi64x(m3, m7); \ + b1 = _mm_set_epi64x(m11, m2) +#define LOAD_MSG_7_1(b0, b1) \ + b0 = _mm_set_epi64x(m7, m13); \ + b1 = _mm_set_epi64x(m3, m12) +#define LOAD_MSG_7_2(b0, b1) \ + b0 = _mm_set_epi64x(m14, m11); \ + b1 = _mm_set_epi64x(m9, m1) +#define LOAD_MSG_7_3(b0, b1) \ + b0 = _mm_set_epi64x(m15, m5); \ + b1 = _mm_set_epi64x(m2, m8) +#define LOAD_MSG_7_4(b0, b1) \ + b0 = _mm_set_epi64x(m4, m0); \ + b1 = _mm_set_epi64x(m10, m6) +#define LOAD_MSG_8_1(b0, b1) \ + b0 = _mm_set_epi64x(m14, m6); \ + b1 = _mm_set_epi64x(m0, m11) +#define LOAD_MSG_8_2(b0, b1) \ + b0 = _mm_set_epi64x(m9, m15); \ + b1 = _mm_set_epi64x(m8, m3) +#define LOAD_MSG_8_3(b0, b1) \ + b0 = _mm_set_epi64x(m13, m12); \ + b1 = _mm_set_epi64x(m10, m1) +#define LOAD_MSG_8_4(b0, b1) \ + b0 = _mm_set_epi64x(m7, m2); \ + b1 = _mm_set_epi64x(m5, m4) +#define LOAD_MSG_9_1(b0, b1) \ + b0 = _mm_set_epi64x(m8, m10); \ + b1 = _mm_set_epi64x(m1, m7) +#define LOAD_MSG_9_2(b0, b1) \ + b0 = _mm_set_epi64x(m4, m2); \ + b1 = _mm_set_epi64x(m5, m6) +#define LOAD_MSG_9_3(b0, b1) \ + b0 = _mm_set_epi64x(m9, m15); \ + b1 = _mm_set_epi64x(m13, m3) +#define LOAD_MSG_9_4(b0, b1) \ + b0 = _mm_set_epi64x(m14, m11); \ + b1 = _mm_set_epi64x(m0, m12) +#define LOAD_MSG_10_1(b0, b1) \ + b0 = _mm_set_epi64x(m2, m0); \ + b1 = _mm_set_epi64x(m6, m4) +#define LOAD_MSG_10_2(b0, b1) \ + b0 = _mm_set_epi64x(m3, m1); \ + b1 = _mm_set_epi64x(m7, m5) +#define LOAD_MSG_10_3(b0, b1) \ + b0 = _mm_set_epi64x(m10, m8); \ + b1 = _mm_set_epi64x(m14, m12) +#define LOAD_MSG_10_4(b0, b1) \ + b0 = _mm_set_epi64x(m11, m9); \ + b1 = _mm_set_epi64x(m15, m13) +#define LOAD_MSG_11_1(b0, b1) \ + b0 = _mm_set_epi64x(m4, m14); \ + b1 = _mm_set_epi64x(m13, m9) +#define LOAD_MSG_11_2(b0, b1) \ + b0 = _mm_set_epi64x(m8, m10); \ + b1 = _mm_set_epi64x(m6, m15) +#define LOAD_MSG_11_3(b0, b1) \ + b0 = _mm_set_epi64x(m0, m1); \ + b1 = _mm_set_epi64x(m5, m11) +#define LOAD_MSG_11_4(b0, b1) \ + b0 = _mm_set_epi64x(m2, m12); \ + b1 = _mm_set_epi64x(m3, m7) + +#endif diff --git a/sodium/sodium_blake2b-load-sse41.h b/sodium/sodium_blake2b-load-sse41.h new file mode 100644 index 00000000..31745fc1 --- /dev/null +++ b/sodium/sodium_blake2b-load-sse41.h @@ -0,0 +1,307 @@ +/* + BLAKE2 reference source code package - optimized C implementations + + Written in 2012 by Samuel Neves + + To the extent possible under law, the author(s) have dedicated all copyright + and related and neighboring rights to this software to the public domain + worldwide. This software is distributed without any warranty. + + You should have received a copy of the CC0 Public Domain Dedication along + with + this software. If not, see + . +*/ + +#ifndef blake2b_load_sse41_H +#define blake2b_load_sse41_H + +#define LOAD_MSG_0_1(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m0, m1); \ + b1 = _mm_unpacklo_epi64(m2, m3); \ + } while (0) + +#define LOAD_MSG_0_2(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m0, m1); \ + b1 = _mm_unpackhi_epi64(m2, m3); \ + } while (0) + +#define LOAD_MSG_0_3(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m4, m5); \ + b1 = _mm_unpacklo_epi64(m6, m7); \ + } while (0) + +#define LOAD_MSG_0_4(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m4, m5); \ + b1 = _mm_unpackhi_epi64(m6, m7); \ + } while (0) + +#define LOAD_MSG_1_1(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m7, m2); \ + b1 = _mm_unpackhi_epi64(m4, m6); \ + } while (0) + +#define LOAD_MSG_1_2(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m5, m4); \ + b1 = _mm_alignr_epi8(m3, m7, 8); \ + } while (0) + +#define LOAD_MSG_1_3(b0, b1) \ + do { \ + b0 = _mm_shuffle_epi32(m0, _MM_SHUFFLE(1, 0, 3, 2)); \ + b1 = _mm_unpackhi_epi64(m5, m2); \ + } while (0) + +#define LOAD_MSG_1_4(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m6, m1); \ + b1 = _mm_unpackhi_epi64(m3, m1); \ + } while (0) + +#define LOAD_MSG_2_1(b0, b1) \ + do { \ + b0 = _mm_alignr_epi8(m6, m5, 8); \ + b1 = _mm_unpackhi_epi64(m2, m7); \ + } while (0) + +#define LOAD_MSG_2_2(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m4, m0); \ + b1 = _mm_blend_epi16(m1, m6, 0xF0); \ + } while (0) + +#define LOAD_MSG_2_3(b0, b1) \ + do { \ + b0 = _mm_blend_epi16(m5, m1, 0xF0); \ + b1 = _mm_unpackhi_epi64(m3, m4); \ + } while (0) + +#define LOAD_MSG_2_4(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m7, m3); \ + b1 = _mm_alignr_epi8(m2, m0, 8); \ + } while (0) + +#define LOAD_MSG_3_1(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m3, m1); \ + b1 = _mm_unpackhi_epi64(m6, m5); \ + } while (0) + +#define LOAD_MSG_3_2(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m4, m0); \ + b1 = _mm_unpacklo_epi64(m6, m7); \ + } while (0) + +#define LOAD_MSG_3_3(b0, b1) \ + do { \ + b0 = _mm_blend_epi16(m1, m2, 0xF0); \ + b1 = _mm_blend_epi16(m2, m7, 0xF0); \ + } while (0) + +#define LOAD_MSG_3_4(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m3, m5); \ + b1 = _mm_unpacklo_epi64(m0, m4); \ + } while (0) + +#define LOAD_MSG_4_1(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m4, m2); \ + b1 = _mm_unpacklo_epi64(m1, m5); \ + } while (0) + +#define LOAD_MSG_4_2(b0, b1) \ + do { \ + b0 = _mm_blend_epi16(m0, m3, 0xF0); \ + b1 = _mm_blend_epi16(m2, m7, 0xF0); \ + } while (0) + +#define LOAD_MSG_4_3(b0, b1) \ + do { \ + b0 = _mm_blend_epi16(m7, m5, 0xF0); \ + b1 = _mm_blend_epi16(m3, m1, 0xF0); \ + } while (0) + +#define LOAD_MSG_4_4(b0, b1) \ + do { \ + b0 = _mm_alignr_epi8(m6, m0, 8); \ + b1 = _mm_blend_epi16(m4, m6, 0xF0); \ + } while (0) + +#define LOAD_MSG_5_1(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m1, m3); \ + b1 = _mm_unpacklo_epi64(m0, m4); \ + } while (0) + +#define LOAD_MSG_5_2(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m6, m5); \ + b1 = _mm_unpackhi_epi64(m5, m1); \ + } while (0) + +#define LOAD_MSG_5_3(b0, b1) \ + do { \ + b0 = _mm_blend_epi16(m2, m3, 0xF0); \ + b1 = _mm_unpackhi_epi64(m7, m0); \ + } while (0) + +#define LOAD_MSG_5_4(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m6, m2); \ + b1 = _mm_blend_epi16(m7, m4, 0xF0); \ + } while (0) + +#define LOAD_MSG_6_1(b0, b1) \ + do { \ + b0 = _mm_blend_epi16(m6, m0, 0xF0); \ + b1 = _mm_unpacklo_epi64(m7, m2); \ + } while (0) + +#define LOAD_MSG_6_2(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m2, m7); \ + b1 = _mm_alignr_epi8(m5, m6, 8); \ + } while (0) + +#define LOAD_MSG_6_3(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m0, m3); \ + b1 = _mm_shuffle_epi32(m4, _MM_SHUFFLE(1, 0, 3, 2)); \ + } while (0) + +#define LOAD_MSG_6_4(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m3, m1); \ + b1 = _mm_blend_epi16(m1, m5, 0xF0); \ + } while (0) + +#define LOAD_MSG_7_1(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m6, m3); \ + b1 = _mm_blend_epi16(m6, m1, 0xF0); \ + } while (0) + +#define LOAD_MSG_7_2(b0, b1) \ + do { \ + b0 = _mm_alignr_epi8(m7, m5, 8); \ + b1 = _mm_unpackhi_epi64(m0, m4); \ + } while (0) + +#define LOAD_MSG_7_3(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m2, m7); \ + b1 = _mm_unpacklo_epi64(m4, m1); \ + } while (0) + +#define LOAD_MSG_7_4(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m0, m2); \ + b1 = _mm_unpacklo_epi64(m3, m5); \ + } while (0) + +#define LOAD_MSG_8_1(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m3, m7); \ + b1 = _mm_alignr_epi8(m0, m5, 8); \ + } while (0) + +#define LOAD_MSG_8_2(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m7, m4); \ + b1 = _mm_alignr_epi8(m4, m1, 8); \ + } while (0) + +#define LOAD_MSG_8_3(b0, b1) \ + do { \ + b0 = m6; \ + b1 = _mm_alignr_epi8(m5, m0, 8); \ + } while (0) + +#define LOAD_MSG_8_4(b0, b1) \ + do { \ + b0 = _mm_blend_epi16(m1, m3, 0xF0); \ + b1 = m2; \ + } while (0) + +#define LOAD_MSG_9_1(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m5, m4); \ + b1 = _mm_unpackhi_epi64(m3, m0); \ + } while (0) + +#define LOAD_MSG_9_2(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m1, m2); \ + b1 = _mm_blend_epi16(m3, m2, 0xF0); \ + } while (0) + +#define LOAD_MSG_9_3(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m7, m4); \ + b1 = _mm_unpackhi_epi64(m1, m6); \ + } while (0) + +#define LOAD_MSG_9_4(b0, b1) \ + do { \ + b0 = _mm_alignr_epi8(m7, m5, 8); \ + b1 = _mm_unpacklo_epi64(m6, m0); \ + } while (0) + +#define LOAD_MSG_10_1(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m0, m1); \ + b1 = _mm_unpacklo_epi64(m2, m3); \ + } while (0) + +#define LOAD_MSG_10_2(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m0, m1); \ + b1 = _mm_unpackhi_epi64(m2, m3); \ + } while (0) + +#define LOAD_MSG_10_3(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m4, m5); \ + b1 = _mm_unpacklo_epi64(m6, m7); \ + } while (0) + +#define LOAD_MSG_10_4(b0, b1) \ + do { \ + b0 = _mm_unpackhi_epi64(m4, m5); \ + b1 = _mm_unpackhi_epi64(m6, m7); \ + } while (0) + +#define LOAD_MSG_11_1(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m7, m2); \ + b1 = _mm_unpackhi_epi64(m4, m6); \ + } while (0) + +#define LOAD_MSG_11_2(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m5, m4); \ + b1 = _mm_alignr_epi8(m3, m7, 8); \ + } while (0) + +#define LOAD_MSG_11_3(b0, b1) \ + do { \ + b0 = _mm_shuffle_epi32(m0, _MM_SHUFFLE(1, 0, 3, 2)); \ + b1 = _mm_unpackhi_epi64(m5, m2); \ + } while (0) + +#define LOAD_MSG_11_4(b0, b1) \ + do { \ + b0 = _mm_unpacklo_epi64(m6, m1); \ + b1 = _mm_unpackhi_epi64(m3, m1); \ + } while (0) + +#endif diff --git a/sodium/sodium_blake2b-ref.c b/sodium/sodium_blake2b-ref.c new file mode 100644 index 00000000..518a415b --- /dev/null +++ b/sodium/sodium_blake2b-ref.c @@ -0,0 +1,453 @@ +/* + BLAKE2 reference source code package - C implementations + + Written in 2012 by Samuel Neves + + To the extent possible under law, the author(s) have dedicated all copyright + and related and neighboring rights to this software to the public domain + worldwide. This software is distributed without any warranty. + + You should have received a copy of the CC0 Public Domain Dedication along + with + this software. If not, see + . +*/ + +#include +#include +#include +#include +#include +#include + +#include "sodium_blake2.h" +#include "sodium_core.h" +#include "sodium_private_common.h" +#include "sodium_runtime.h" +#include "sodium_utils.h" + +static blake2b_compress_fn blake2b_compress = blake2b_compress_ref; + +static const uint64_t blake2b_IV[8] = { + 0x6a09e667f3bcc908ULL, 0xbb67ae8584caa73bULL, 0x3c6ef372fe94f82bULL, + 0xa54ff53a5f1d36f1ULL, 0x510e527fade682d1ULL, 0x9b05688c2b3e6c1fULL, + 0x1f83d9abfb41bd6bULL, 0x5be0cd19137e2179ULL +}; + +/* LCOV_EXCL_START */ +static inline int +blake2b_set_lastnode(blake2b_state *S) +{ + S->f[1] = (uint64_t) -1; + return 0; +} +/* LCOV_EXCL_STOP */ + +static inline int +blake2b_is_lastblock(const blake2b_state *S) +{ + return S->f[0] != 0; +} + +static inline int +blake2b_set_lastblock(blake2b_state *S) +{ + if (S->last_node) { + blake2b_set_lastnode(S); + } + S->f[0] = (uint64_t) -1; + return 0; +} + +static inline int +blake2b_increment_counter(blake2b_state *S, const uint64_t inc) +{ +#ifdef HAVE_TI_MODE + uint128_t t = ((uint128_t) S->t[1] << 64) | S->t[0]; + t += inc; + S->t[0] = (uint64_t)(t >> 0); + S->t[1] = (uint64_t)(t >> 64); +#else + S->t[0] += inc; + S->t[1] += (S->t[0] < inc); +#endif + return 0; +} + +/* Parameter-related functions */ +static inline int +blake2b_param_set_salt(blake2b_param *P, const uint8_t salt[BLAKE2B_SALTBYTES]) +{ + memcpy(P->salt, salt, BLAKE2B_SALTBYTES); + return 0; +} + +static inline int +blake2b_param_set_personal(blake2b_param *P, + const uint8_t personal[BLAKE2B_PERSONALBYTES]) +{ + memcpy(P->personal, personal, BLAKE2B_PERSONALBYTES); + return 0; +} + +static inline int +blake2b_init0(blake2b_state *S) +{ + int i; + + for (i = 0; i < 8; i++) { + S->h[i] = blake2b_IV[i]; + } + /* zero everything between .t and .last_node */ + memset((void *) &S->t, 0, + offsetof(blake2b_state, last_node) + sizeof(S->last_node) + - offsetof(blake2b_state, t)); + return 0; +} + +/* init xors IV with input parameter block */ +int +blake2b_init_param(blake2b_state *S, const blake2b_param *P) +{ + size_t i; + const uint8_t *p; + + COMPILER_ASSERT(sizeof *P == 64); + blake2b_init0(S); + p = (const uint8_t *) (P); + + /* IV XOR ParamBlock */ + for (i = 0; i < 8; i++) { + S->h[i] ^= LOAD64_LE(p + sizeof(S->h[i]) * i); + } + return 0; +} + +int +blake2b_init(blake2b_state *S, const uint8_t outlen) +{ + blake2b_param P[1]; + + if ((!outlen) || (outlen > BLAKE2B_OUTBYTES)) { + sodium_misuse(); + } + P->digest_length = outlen; + P->key_length = 0; + P->fanout = 1; + P->depth = 1; + STORE32_LE(P->leaf_length, 0); + STORE64_LE(P->node_offset, 0); + P->node_depth = 0; + P->inner_length = 0; + memset(P->reserved, 0, sizeof(P->reserved)); + memset(P->salt, 0, sizeof(P->salt)); + memset(P->personal, 0, sizeof(P->personal)); + return blake2b_init_param(S, P); +} + +int +blake2b_init_salt_personal(blake2b_state *S, const uint8_t outlen, + const void *salt, const void *personal) +{ + blake2b_param P[1]; + + if ((!outlen) || (outlen > BLAKE2B_OUTBYTES)) { + sodium_misuse(); + } + P->digest_length = outlen; + P->key_length = 0; + P->fanout = 1; + P->depth = 1; + STORE32_LE(P->leaf_length, 0); + STORE64_LE(P->node_offset, 0); + P->node_depth = 0; + P->inner_length = 0; + memset(P->reserved, 0, sizeof(P->reserved)); + if (salt != NULL) { + blake2b_param_set_salt(P, (const uint8_t *) salt); + } else { + memset(P->salt, 0, sizeof(P->salt)); + } + if (personal != NULL) { + blake2b_param_set_personal(P, (const uint8_t *) personal); + } else { + memset(P->personal, 0, sizeof(P->personal)); + } + return blake2b_init_param(S, P); +} + +int +blake2b_init_key(blake2b_state *S, const uint8_t outlen, const void *key, + const uint8_t keylen) +{ + blake2b_param P[1]; + + if ((!outlen) || (outlen > BLAKE2B_OUTBYTES)) { + sodium_misuse(); + } + if (!key || !keylen || keylen > BLAKE2B_KEYBYTES) { + sodium_misuse(); /* does not return */ + } + P->digest_length = outlen; + P->key_length = keylen; + P->fanout = 1; + P->depth = 1; + STORE32_LE(P->leaf_length, 0); + STORE64_LE(P->node_offset, 0); + P->node_depth = 0; + P->inner_length = 0; + memset(P->reserved, 0, sizeof(P->reserved)); + memset(P->salt, 0, sizeof(P->salt)); + memset(P->personal, 0, sizeof(P->personal)); + + if (blake2b_init_param(S, P) < 0) { + sodium_misuse(); + } + { + uint8_t block[BLAKE2B_BLOCKBYTES]; + memset(block, 0, BLAKE2B_BLOCKBYTES); + memcpy(block, key, keylen); /* key and keylen cannot be 0 */ + blake2b_update(S, block, BLAKE2B_BLOCKBYTES); + sodium_memzero(block, BLAKE2B_BLOCKBYTES); /* Burn the key from stack */ + } + return 0; +} + +int +blake2b_init_key_salt_personal(blake2b_state *S, const uint8_t outlen, + const void *key, const uint8_t keylen, + const void *salt, const void *personal) +{ + blake2b_param P[1]; + + if ((!outlen) || (outlen > BLAKE2B_OUTBYTES)) { + sodium_misuse(); + } + if (!key || !keylen || keylen > BLAKE2B_KEYBYTES) { + sodium_misuse(); /* does not return */ + } + P->digest_length = outlen; + P->key_length = keylen; + P->fanout = 1; + P->depth = 1; + STORE32_LE(P->leaf_length, 0); + STORE64_LE(P->node_offset, 0); + P->node_depth = 0; + P->inner_length = 0; + memset(P->reserved, 0, sizeof(P->reserved)); + if (salt != NULL) { + blake2b_param_set_salt(P, (const uint8_t *) salt); + } else { + memset(P->salt, 0, sizeof(P->salt)); + } + if (personal != NULL) { + blake2b_param_set_personal(P, (const uint8_t *) personal); + } else { + memset(P->personal, 0, sizeof(P->personal)); + } + + if (blake2b_init_param(S, P) < 0) { + sodium_misuse(); + } + { + uint8_t block[BLAKE2B_BLOCKBYTES]; + memset(block, 0, BLAKE2B_BLOCKBYTES); + memcpy(block, key, keylen); /* key and keylen cannot be 0 */ + blake2b_update(S, block, BLAKE2B_BLOCKBYTES); + sodium_memzero(block, BLAKE2B_BLOCKBYTES); /* Burn the key from stack */ + } + return 0; +} + +/* inlen now in bytes */ +int +blake2b_update(blake2b_state *S, const uint8_t *in, uint64_t inlen) +{ + while (inlen > 0) { + size_t left = S->buflen; + size_t fill = 2 * BLAKE2B_BLOCKBYTES - left; + + if (inlen > fill) { + memcpy(S->buf + left, in, fill); /* Fill buffer */ + S->buflen += fill; + blake2b_increment_counter(S, BLAKE2B_BLOCKBYTES); + blake2b_compress(S, S->buf); /* Compress */ + memcpy(S->buf, S->buf + BLAKE2B_BLOCKBYTES, + BLAKE2B_BLOCKBYTES); /* Shift buffer left */ + S->buflen -= BLAKE2B_BLOCKBYTES; + in += fill; + inlen -= fill; + } else /* inlen <= fill */ + { + memcpy(S->buf + left, in, inlen); + S->buflen += inlen; /* Be lazy, do not compress */ + in += inlen; + inlen -= inlen; + } + } + + return 0; +} + +int +blake2b_final(blake2b_state *S, uint8_t *out, uint8_t outlen) +{ + unsigned char buffer[BLAKE2B_OUTBYTES]; + + if (!outlen || outlen > BLAKE2B_OUTBYTES) { + sodium_misuse(); + } + if (blake2b_is_lastblock(S)) { + return -1; + } + if (S->buflen > BLAKE2B_BLOCKBYTES) { + blake2b_increment_counter(S, BLAKE2B_BLOCKBYTES); + blake2b_compress(S, S->buf); + S->buflen -= BLAKE2B_BLOCKBYTES; + assert(S->buflen <= BLAKE2B_BLOCKBYTES); + memcpy(S->buf, S->buf + BLAKE2B_BLOCKBYTES, S->buflen); + } + + blake2b_increment_counter(S, S->buflen); + blake2b_set_lastblock(S); + memset(S->buf + S->buflen, 0, + 2 * BLAKE2B_BLOCKBYTES - S->buflen); /* Padding */ + blake2b_compress(S, S->buf); + + COMPILER_ASSERT(sizeof buffer == 64U); + STORE64_LE(buffer + 8 * 0, S->h[0]); + STORE64_LE(buffer + 8 * 1, S->h[1]); + STORE64_LE(buffer + 8 * 2, S->h[2]); + STORE64_LE(buffer + 8 * 3, S->h[3]); + STORE64_LE(buffer + 8 * 4, S->h[4]); + STORE64_LE(buffer + 8 * 5, S->h[5]); + STORE64_LE(buffer + 8 * 6, S->h[6]); + STORE64_LE(buffer + 8 * 7, S->h[7]); + memcpy(out, buffer, outlen); /* outlen <= BLAKE2B_OUTBYTES (64) */ + + sodium_memzero(S->h, sizeof S->h); + sodium_memzero(S->buf, sizeof S->buf); + + return 0; +} + +/* inlen, at least, should be uint64_t. Others can be size_t. */ +int +blake2b(uint8_t *out, const void *in, const void *key, const uint8_t outlen, + const uint64_t inlen, uint8_t keylen) +{ + CRYPTO_ALIGN(64) blake2b_state S[1]; + + /* Verify parameters */ + if (NULL == in && inlen > 0) { + sodium_misuse(); + } + if (NULL == out) { + sodium_misuse(); + } + if (!outlen || outlen > BLAKE2B_OUTBYTES) { + sodium_misuse(); + } + if (NULL == key && keylen > 0) { + sodium_misuse(); + } + if (keylen > BLAKE2B_KEYBYTES) { + sodium_misuse(); + } + if (keylen > 0) { + if (blake2b_init_key(S, outlen, key, keylen) < 0) { + sodium_misuse(); + } + } else { + if (blake2b_init(S, outlen) < 0) { + sodium_misuse(); + } + } + + blake2b_update(S, (const uint8_t *) in, inlen); + blake2b_final(S, out, outlen); + return 0; +} + +int +blake2b_salt_personal(uint8_t *out, const void *in, const void *key, + const uint8_t outlen, const uint64_t inlen, + uint8_t keylen, const void *salt, const void *personal) +{ + CRYPTO_ALIGN(64) blake2b_state S[1]; + + /* Verify parameters */ + if (NULL == in && inlen > 0) { + sodium_misuse(); + } + if (NULL == out) { + sodium_misuse(); + } + if (!outlen || outlen > BLAKE2B_OUTBYTES) { + sodium_misuse(); + } + if (NULL == key && keylen > 0) { + sodium_misuse(); + } + if (keylen > BLAKE2B_KEYBYTES) { + sodium_misuse(); + } + if (keylen > 0) { + if (blake2b_init_key_salt_personal(S, outlen, key, keylen, salt, + personal) < 0) { + sodium_misuse(); + } + } else { + if (blake2b_init_salt_personal(S, outlen, salt, personal) < 0) { + sodium_misuse(); + } + } + + blake2b_update(S, (const uint8_t *) in, inlen); + blake2b_final(S, out, outlen); + return 0; +} + +int +blake2b_pick_best_implementation(void) +{ +/* LCOV_EXCL_START */ +#if defined(HAVE_AVX2INTRIN_H) && defined(HAVE_TMMINTRIN_H) && \ + defined(HAVE_SMMINTRIN_H) + if (sodium_runtime_has_avx2()) { + #if NETCODE_CRYPTO_LOGS + printf( "blake2b -> avx2\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + blake2b_compress = blake2b_compress_avx2; + return 0; + } +#endif +#if defined(HAVE_EMMINTRIN_H) && defined(HAVE_TMMINTRIN_H) && \ + defined(HAVE_SMMINTRIN_H) + if (sodium_runtime_has_sse41()) { + #if NETCODE_CRYPTO_LOGS + printf( "blake2b -> sse41\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + blake2b_compress = blake2b_compress_sse41; + return 0; + } +#endif +#if defined(HAVE_EMMINTRIN_H) && defined(HAVE_TMMINTRIN_H) + if (sodium_runtime_has_ssse3()) { + #if NETCODE_CRYPTO_LOGS + printf( "blake2b -> ssse3\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + blake2b_compress = blake2b_compress_ssse3; + return 0; + } +#endif + + #if NETCODE_CRYPTO_LOGS + printf( "blake2b -> ref\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + + blake2b_compress = blake2b_compress_ref; + + return 0; + /* LCOV_EXCL_STOP */ +} diff --git a/sodium/sodium_box_curve25519xchacha20poly1305.c b/sodium/sodium_box_curve25519xchacha20poly1305.c new file mode 100644 index 00000000..91d69120 --- /dev/null +++ b/sodium/sodium_box_curve25519xchacha20poly1305.c @@ -0,0 +1,204 @@ + +#include +#include +#include +#include + +#include "sodium_core.h" +#include "sodium_crypto_box_curve25519xchacha20poly1305.h" +#include "sodium_crypto_core_hchacha20.h" +#include "sodium_crypto_hash_sha512.h" +#include "sodium_crypto_scalarmult_curve25519.h" +#include "sodium_crypto_secretbox_xchacha20poly1305.h" +#include "sodium_private_common.h" +#include "sodium_randombytes.h" +#include "sodium_utils.h" + +int +crypto_box_curve25519xchacha20poly1305_seed_keypair(unsigned char *pk, + unsigned char *sk, + const unsigned char *seed) +{ + unsigned char hash[64]; + + crypto_hash_sha512(hash, seed, 32); + memcpy(sk, hash, 32); + sodium_memzero(hash, sizeof hash); + + return crypto_scalarmult_curve25519_base(pk, sk); +} + +int +crypto_box_curve25519xchacha20poly1305_keypair(unsigned char *pk, + unsigned char *sk) +{ + randombytes_buf(sk, 32); + + return crypto_scalarmult_curve25519_base(pk, sk); +} + +int +crypto_box_curve25519xchacha20poly1305_beforenm(unsigned char *k, + const unsigned char *pk, + const unsigned char *sk) +{ + static const unsigned char zero[16] = { 0 }; + unsigned char s[32]; + + if (crypto_scalarmult_curve25519(s, sk, pk) != 0) { + return -1; + } + return crypto_core_hchacha20(k, zero, s, NULL); +} + +int +crypto_box_curve25519xchacha20poly1305_detached_afternm( + unsigned char *c, unsigned char *mac, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, const unsigned char *k) +{ + return crypto_secretbox_xchacha20poly1305_detached(c, mac, m, mlen, n, k); +} + +int +crypto_box_curve25519xchacha20poly1305_detached( + unsigned char *c, unsigned char *mac, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, const unsigned char *pk, + const unsigned char *sk) +{ + unsigned char k[crypto_box_curve25519xchacha20poly1305_BEFORENMBYTES]; + int ret; + + COMPILER_ASSERT(crypto_box_curve25519xchacha20poly1305_BEFORENMBYTES >= + crypto_secretbox_xchacha20poly1305_KEYBYTES); + if (crypto_box_curve25519xchacha20poly1305_beforenm(k, pk, sk) != 0) { + return -1; + } + ret = crypto_box_curve25519xchacha20poly1305_detached_afternm(c, mac, m, + mlen, n, k); + sodium_memzero(k, sizeof k); + + return ret; +} + +int +crypto_box_curve25519xchacha20poly1305_easy_afternm(unsigned char *c, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, + const unsigned char *k) +{ + if (mlen > crypto_box_curve25519xchacha20poly1305_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return crypto_box_curve25519xchacha20poly1305_detached_afternm( + c + crypto_box_curve25519xchacha20poly1305_MACBYTES, c, m, mlen, n, k); +} + +int +crypto_box_curve25519xchacha20poly1305_easy( + unsigned char *c, const unsigned char *m, unsigned long long mlen, + const unsigned char *n, const unsigned char *pk, const unsigned char *sk) +{ + if (mlen > crypto_box_curve25519xchacha20poly1305_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return crypto_box_curve25519xchacha20poly1305_detached( + c + crypto_box_curve25519xchacha20poly1305_MACBYTES, c, m, mlen, n, pk, + sk); +} + +int +crypto_box_curve25519xchacha20poly1305_open_detached_afternm( + unsigned char *m, const unsigned char *c, const unsigned char *mac, + unsigned long long clen, const unsigned char *n, const unsigned char *k) +{ + return crypto_secretbox_xchacha20poly1305_open_detached(m, c, mac, clen, n, + k); +} + +int +crypto_box_curve25519xchacha20poly1305_open_detached( + unsigned char *m, const unsigned char *c, const unsigned char *mac, + unsigned long long clen, const unsigned char *n, const unsigned char *pk, + const unsigned char *sk) +{ + unsigned char k[crypto_box_curve25519xchacha20poly1305_BEFORENMBYTES]; + int ret; + + if (crypto_box_curve25519xchacha20poly1305_beforenm(k, pk, sk) != 0) { + return -1; + } + ret = crypto_box_curve25519xchacha20poly1305_open_detached_afternm( + m, c, mac, clen, n, k); + sodium_memzero(k, sizeof k); + + return ret; +} + +int +crypto_box_curve25519xchacha20poly1305_open_easy_afternm( + unsigned char *m, const unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k) +{ + if (clen < crypto_box_curve25519xchacha20poly1305_MACBYTES) { + return -1; + } + return crypto_box_curve25519xchacha20poly1305_open_detached_afternm( + m, c + crypto_box_curve25519xchacha20poly1305_MACBYTES, c, + clen - crypto_box_curve25519xchacha20poly1305_MACBYTES, n, k); +} + +int +crypto_box_curve25519xchacha20poly1305_open_easy( + unsigned char *m, const unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *pk, const unsigned char *sk) +{ + if (clen < crypto_box_curve25519xchacha20poly1305_MACBYTES) { + return -1; + } + return crypto_box_curve25519xchacha20poly1305_open_detached( + m, c + crypto_box_curve25519xchacha20poly1305_MACBYTES, c, + clen - crypto_box_curve25519xchacha20poly1305_MACBYTES, n, pk, sk); +} + +size_t +crypto_box_curve25519xchacha20poly1305_seedbytes(void) +{ + return crypto_box_curve25519xchacha20poly1305_SEEDBYTES; +} + +size_t +crypto_box_curve25519xchacha20poly1305_publickeybytes(void) +{ + return crypto_box_curve25519xchacha20poly1305_PUBLICKEYBYTES; +} + +size_t +crypto_box_curve25519xchacha20poly1305_secretkeybytes(void) +{ + return crypto_box_curve25519xchacha20poly1305_SECRETKEYBYTES; +} + +size_t +crypto_box_curve25519xchacha20poly1305_beforenmbytes(void) +{ + return crypto_box_curve25519xchacha20poly1305_BEFORENMBYTES; +} + +size_t +crypto_box_curve25519xchacha20poly1305_noncebytes(void) +{ + return crypto_box_curve25519xchacha20poly1305_NONCEBYTES; +} + +size_t +crypto_box_curve25519xchacha20poly1305_macbytes(void) +{ + return crypto_box_curve25519xchacha20poly1305_MACBYTES; +} + +size_t +crypto_box_curve25519xchacha20poly1305_messagebytes_max(void) +{ + return crypto_box_curve25519xchacha20poly1305_MESSAGEBYTES_MAX; +} diff --git a/sodium/sodium_box_curve25519xsalsa20poly1305.c b/sodium/sodium_box_curve25519xsalsa20poly1305.c new file mode 100644 index 00000000..12327a44 --- /dev/null +++ b/sodium/sodium_box_curve25519xsalsa20poly1305.c @@ -0,0 +1,156 @@ +#include + +#include "sodium_crypto_box_curve25519xsalsa20poly1305.h" +#include "sodium_crypto_core_hsalsa20.h" +#include "sodium_crypto_hash_sha512.h" +#include "sodium_crypto_scalarmult_curve25519.h" +#include "sodium_crypto_secretbox_xsalsa20poly1305.h" +#include "sodium_randombytes.h" +#include "sodium_utils.h" + +int +crypto_box_curve25519xsalsa20poly1305_seed_keypair(unsigned char *pk, + unsigned char *sk, + const unsigned char *seed) +{ + unsigned char hash[64]; + + crypto_hash_sha512(hash, seed, 32); + memcpy(sk, hash, 32); + sodium_memzero(hash, sizeof hash); + + return crypto_scalarmult_curve25519_base(pk, sk); +} + +int +crypto_box_curve25519xsalsa20poly1305_keypair(unsigned char *pk, + unsigned char *sk) +{ + randombytes_buf(sk, 32); + + return crypto_scalarmult_curve25519_base(pk, sk); +} + +int +crypto_box_curve25519xsalsa20poly1305_beforenm(unsigned char *k, + const unsigned char *pk, + const unsigned char *sk) +{ + static const unsigned char zero[16] = { 0 }; + unsigned char s[32]; + + if (crypto_scalarmult_curve25519(s, sk, pk) != 0) { + return -1; + } + return crypto_core_hsalsa20(k, zero, s, NULL); +} + +int +crypto_box_curve25519xsalsa20poly1305_afternm(unsigned char *c, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, + const unsigned char *k) +{ + return crypto_secretbox_xsalsa20poly1305(c, m, mlen, n, k); +} + +int +crypto_box_curve25519xsalsa20poly1305_open_afternm(unsigned char *m, + const unsigned char *c, + unsigned long long clen, + const unsigned char *n, + const unsigned char *k) +{ + return crypto_secretbox_xsalsa20poly1305_open(m, c, clen, n, k); +} + +int +crypto_box_curve25519xsalsa20poly1305(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, + const unsigned char *pk, + const unsigned char *sk) +{ + unsigned char k[crypto_box_curve25519xsalsa20poly1305_BEFORENMBYTES]; + int ret; + + if (crypto_box_curve25519xsalsa20poly1305_beforenm(k, pk, sk) != 0) { + return -1; + } + ret = crypto_box_curve25519xsalsa20poly1305_afternm(c, m, mlen, n, k); + sodium_memzero(k, sizeof k); + + return ret; +} + +int +crypto_box_curve25519xsalsa20poly1305_open( + unsigned char *m, const unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *pk, const unsigned char *sk) +{ + unsigned char k[crypto_box_curve25519xsalsa20poly1305_BEFORENMBYTES]; + int ret; + + if (crypto_box_curve25519xsalsa20poly1305_beforenm(k, pk, sk) != 0) { + return -1; + } + ret = crypto_box_curve25519xsalsa20poly1305_open_afternm(m, c, clen, n, k); + sodium_memzero(k, sizeof k); + + return ret; +} + +size_t +crypto_box_curve25519xsalsa20poly1305_seedbytes(void) +{ + return crypto_box_curve25519xsalsa20poly1305_SEEDBYTES; +} + +size_t +crypto_box_curve25519xsalsa20poly1305_publickeybytes(void) +{ + return crypto_box_curve25519xsalsa20poly1305_PUBLICKEYBYTES; +} + +size_t +crypto_box_curve25519xsalsa20poly1305_secretkeybytes(void) +{ + return crypto_box_curve25519xsalsa20poly1305_SECRETKEYBYTES; +} + +size_t +crypto_box_curve25519xsalsa20poly1305_beforenmbytes(void) +{ + return crypto_box_curve25519xsalsa20poly1305_BEFORENMBYTES; +} + +size_t +crypto_box_curve25519xsalsa20poly1305_noncebytes(void) +{ + return crypto_box_curve25519xsalsa20poly1305_NONCEBYTES; +} + +size_t +crypto_box_curve25519xsalsa20poly1305_zerobytes(void) +{ + return crypto_box_curve25519xsalsa20poly1305_ZEROBYTES; +} + +size_t +crypto_box_curve25519xsalsa20poly1305_boxzerobytes(void) +{ + return crypto_box_curve25519xsalsa20poly1305_BOXZEROBYTES; +} + +size_t +crypto_box_curve25519xsalsa20poly1305_macbytes(void) +{ + return crypto_box_curve25519xsalsa20poly1305_MACBYTES; +} + +size_t +crypto_box_curve25519xsalsa20poly1305_messagebytes_max(void) +{ + return crypto_box_curve25519xsalsa20poly1305_MESSAGEBYTES_MAX; +} diff --git a/sodium/sodium_box_seal_curve25519xchacha20poly1305.c b/sodium/sodium_box_seal_curve25519xchacha20poly1305.c new file mode 100644 index 00000000..6b1422b0 --- /dev/null +++ b/sodium/sodium_box_seal_curve25519xchacha20poly1305.c @@ -0,0 +1,79 @@ + +#include + +#include "sodium_crypto_box_curve25519xchacha20poly1305.h" +#include "sodium_crypto_generichash.h" +#include "sodium_private_common.h" +#include "sodium_utils.h" + +static int +_crypto_box_curve25519xchacha20poly1305_seal_nonce(unsigned char *nonce, + const unsigned char *pk1, + const unsigned char *pk2) +{ + crypto_generichash_state st; + + crypto_generichash_init(&st, NULL, 0U, + crypto_box_curve25519xchacha20poly1305_NONCEBYTES); + crypto_generichash_update(&st, pk1, + crypto_box_curve25519xchacha20poly1305_PUBLICKEYBYTES); + crypto_generichash_update(&st, pk2, + crypto_box_curve25519xchacha20poly1305_PUBLICKEYBYTES); + crypto_generichash_final(&st, nonce, + crypto_box_curve25519xchacha20poly1305_NONCEBYTES); + + return 0; +} + +int +crypto_box_curve25519xchacha20poly1305_seal(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *pk) +{ + unsigned char nonce[crypto_box_curve25519xchacha20poly1305_NONCEBYTES]; + unsigned char epk[crypto_box_curve25519xchacha20poly1305_PUBLICKEYBYTES]; + unsigned char esk[crypto_box_curve25519xchacha20poly1305_SECRETKEYBYTES]; + int ret; + + if (crypto_box_curve25519xchacha20poly1305_keypair(epk, esk) != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + memcpy(c, epk, crypto_box_curve25519xchacha20poly1305_PUBLICKEYBYTES); + _crypto_box_curve25519xchacha20poly1305_seal_nonce(nonce, epk, pk); + ret = crypto_box_curve25519xchacha20poly1305_easy( + c + crypto_box_curve25519xchacha20poly1305_PUBLICKEYBYTES, m, mlen, + nonce, pk, esk); + sodium_memzero(esk, sizeof esk); + sodium_memzero(epk, sizeof epk); + sodium_memzero(nonce, sizeof nonce); + + return ret; +} + +int +crypto_box_curve25519xchacha20poly1305_seal_open(unsigned char *m, const unsigned char *c, + unsigned long long clen, + const unsigned char *pk, + const unsigned char *sk) +{ + unsigned char nonce[crypto_box_curve25519xchacha20poly1305_NONCEBYTES]; + + if (clen < crypto_box_curve25519xchacha20poly1305_SEALBYTES) { + return -1; + } + _crypto_box_curve25519xchacha20poly1305_seal_nonce(nonce, c, pk); + + COMPILER_ASSERT(crypto_box_curve25519xchacha20poly1305_PUBLICKEYBYTES < + crypto_box_curve25519xchacha20poly1305_SEALBYTES); + + return crypto_box_curve25519xchacha20poly1305_open_easy( + m, c + crypto_box_curve25519xchacha20poly1305_PUBLICKEYBYTES, + clen - crypto_box_curve25519xchacha20poly1305_PUBLICKEYBYTES, + nonce, c, sk); +} + +size_t +crypto_box_curve25519xchacha20poly1305_sealbytes(void) +{ + return crypto_box_curve25519xchacha20poly1305_SEALBYTES; +} diff --git a/sodium/sodium_chacha20-ref.c b/sodium/sodium_chacha20-ref.c new file mode 100644 index 00000000..164d6f32 --- /dev/null +++ b/sodium/sodium_chacha20-ref.c @@ -0,0 +1,312 @@ + +/* + chacha-merged.c version 20080118 + D. J. Bernstein + Public domain. + */ + +#include +#include +#include + +#include "sodium_core.h" +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_private_common.h" +#include "sodium_utils.h" + +#include "sodium_stream_chacha20.h" +#include "sodium_chacha20-ref.h" + +struct chacha_ctx { + uint32_t input[16]; +}; + +typedef struct chacha_ctx chacha_ctx; + +#define U32C(v) (v##U) + +#define U32V(v) ((uint32_t)(v) &U32C(0xFFFFFFFF)) + +#define ROTATE(v, c) (ROTL32(v, c)) +#define XOR(v, w) ((v) ^ (w)) +#define PLUS(v, w) (U32V((v) + (w))) +#define PLUSONE(v) (PLUS((v), 1)) + +#define QUARTERROUND(a, b, c, d) \ + a = PLUS(a, b); \ + d = ROTATE(XOR(d, a), 16); \ + c = PLUS(c, d); \ + b = ROTATE(XOR(b, c), 12); \ + a = PLUS(a, b); \ + d = ROTATE(XOR(d, a), 8); \ + c = PLUS(c, d); \ + b = ROTATE(XOR(b, c), 7); + +static void +chacha_keysetup(chacha_ctx *ctx, const uint8_t *k) +{ + ctx->input[0] = U32C(0x61707865); + ctx->input[1] = U32C(0x3320646e); + ctx->input[2] = U32C(0x79622d32); + ctx->input[3] = U32C(0x6b206574); + ctx->input[4] = LOAD32_LE(k + 0); + ctx->input[5] = LOAD32_LE(k + 4); + ctx->input[6] = LOAD32_LE(k + 8); + ctx->input[7] = LOAD32_LE(k + 12); + ctx->input[8] = LOAD32_LE(k + 16); + ctx->input[9] = LOAD32_LE(k + 20); + ctx->input[10] = LOAD32_LE(k + 24); + ctx->input[11] = LOAD32_LE(k + 28); +} + +static void +chacha_ivsetup(chacha_ctx *ctx, const uint8_t *iv, const uint8_t *counter) +{ + ctx->input[12] = counter == NULL ? 0 : LOAD32_LE(counter + 0); + ctx->input[13] = counter == NULL ? 0 : LOAD32_LE(counter + 4); + ctx->input[14] = LOAD32_LE(iv + 0); + ctx->input[15] = LOAD32_LE(iv + 4); +} + +static void +chacha_ietf_ivsetup(chacha_ctx *ctx, const uint8_t *iv, const uint8_t *counter) +{ + ctx->input[12] = counter == NULL ? 0 : LOAD32_LE(counter); + ctx->input[13] = LOAD32_LE(iv + 0); + ctx->input[14] = LOAD32_LE(iv + 4); + ctx->input[15] = LOAD32_LE(iv + 8); +} + +static void +chacha20_encrypt_bytes(chacha_ctx *ctx, const uint8_t *m, uint8_t *c, + unsigned long long bytes) +{ + uint32_t x0, x1, x2, x3, x4, x5, x6, x7, x8, x9, x10, x11, x12, x13, x14, + x15; + uint32_t j0, j1, j2, j3, j4, j5, j6, j7, j8, j9, j10, j11, j12, j13, j14, + j15; + uint8_t *ctarget = NULL; + uint8_t tmp[64]; + unsigned int i; + + if (!bytes) { + return; /* LCOV_EXCL_LINE */ + } + j0 = ctx->input[0]; + j1 = ctx->input[1]; + j2 = ctx->input[2]; + j3 = ctx->input[3]; + j4 = ctx->input[4]; + j5 = ctx->input[5]; + j6 = ctx->input[6]; + j7 = ctx->input[7]; + j8 = ctx->input[8]; + j9 = ctx->input[9]; + j10 = ctx->input[10]; + j11 = ctx->input[11]; + j12 = ctx->input[12]; + j13 = ctx->input[13]; + j14 = ctx->input[14]; + j15 = ctx->input[15]; + + for (;;) { + if (bytes < 64) { + memset(tmp, 0, 64); + for (i = 0; i < bytes; ++i) { + tmp[i] = m[i]; + } + m = tmp; + ctarget = c; + c = tmp; + } + x0 = j0; + x1 = j1; + x2 = j2; + x3 = j3; + x4 = j4; + x5 = j5; + x6 = j6; + x7 = j7; + x8 = j8; + x9 = j9; + x10 = j10; + x11 = j11; + x12 = j12; + x13 = j13; + x14 = j14; + x15 = j15; + for (i = 20; i > 0; i -= 2) { + QUARTERROUND(x0, x4, x8, x12) + QUARTERROUND(x1, x5, x9, x13) + QUARTERROUND(x2, x6, x10, x14) + QUARTERROUND(x3, x7, x11, x15) + QUARTERROUND(x0, x5, x10, x15) + QUARTERROUND(x1, x6, x11, x12) + QUARTERROUND(x2, x7, x8, x13) + QUARTERROUND(x3, x4, x9, x14) + } + x0 = PLUS(x0, j0); + x1 = PLUS(x1, j1); + x2 = PLUS(x2, j2); + x3 = PLUS(x3, j3); + x4 = PLUS(x4, j4); + x5 = PLUS(x5, j5); + x6 = PLUS(x6, j6); + x7 = PLUS(x7, j7); + x8 = PLUS(x8, j8); + x9 = PLUS(x9, j9); + x10 = PLUS(x10, j10); + x11 = PLUS(x11, j11); + x12 = PLUS(x12, j12); + x13 = PLUS(x13, j13); + x14 = PLUS(x14, j14); + x15 = PLUS(x15, j15); + + x0 = XOR(x0, LOAD32_LE(m + 0)); + x1 = XOR(x1, LOAD32_LE(m + 4)); + x2 = XOR(x2, LOAD32_LE(m + 8)); + x3 = XOR(x3, LOAD32_LE(m + 12)); + x4 = XOR(x4, LOAD32_LE(m + 16)); + x5 = XOR(x5, LOAD32_LE(m + 20)); + x6 = XOR(x6, LOAD32_LE(m + 24)); + x7 = XOR(x7, LOAD32_LE(m + 28)); + x8 = XOR(x8, LOAD32_LE(m + 32)); + x9 = XOR(x9, LOAD32_LE(m + 36)); + x10 = XOR(x10, LOAD32_LE(m + 40)); + x11 = XOR(x11, LOAD32_LE(m + 44)); + x12 = XOR(x12, LOAD32_LE(m + 48)); + x13 = XOR(x13, LOAD32_LE(m + 52)); + x14 = XOR(x14, LOAD32_LE(m + 56)); + x15 = XOR(x15, LOAD32_LE(m + 60)); + + j12 = PLUSONE(j12); + /* LCOV_EXCL_START */ + if (!j12) { + j13 = PLUSONE(j13); + } + /* LCOV_EXCL_STOP */ + + STORE32_LE(c + 0, x0); + STORE32_LE(c + 4, x1); + STORE32_LE(c + 8, x2); + STORE32_LE(c + 12, x3); + STORE32_LE(c + 16, x4); + STORE32_LE(c + 20, x5); + STORE32_LE(c + 24, x6); + STORE32_LE(c + 28, x7); + STORE32_LE(c + 32, x8); + STORE32_LE(c + 36, x9); + STORE32_LE(c + 40, x10); + STORE32_LE(c + 44, x11); + STORE32_LE(c + 48, x12); + STORE32_LE(c + 52, x13); + STORE32_LE(c + 56, x14); + STORE32_LE(c + 60, x15); + + if (bytes <= 64) { + if (bytes < 64) { + for (i = 0; i < (unsigned int) bytes; ++i) { + ctarget[i] = c[i]; /* ctarget cannot be NULL */ + } + } + ctx->input[12] = j12; + ctx->input[13] = j13; + + return; + } + bytes -= 64; + c += 64; + m += 64; + } +} + +static int +stream_ref(unsigned char *c, unsigned long long clen, const unsigned char *n, + const unsigned char *k) +{ + struct chacha_ctx ctx; + + if (!clen) { + return 0; + } + COMPILER_ASSERT(crypto_stream_chacha20_KEYBYTES == 256 / 8); + chacha_keysetup(&ctx, k); + chacha_ivsetup(&ctx, n, NULL); + memset(c, 0, clen); + chacha20_encrypt_bytes(&ctx, c, c, clen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +static int +stream_ietf_ext_ref(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k) +{ + struct chacha_ctx ctx; + + if (!clen) { + return 0; + } + COMPILER_ASSERT(crypto_stream_chacha20_KEYBYTES == 256 / 8); + chacha_keysetup(&ctx, k); + chacha_ietf_ivsetup(&ctx, n, NULL); + memset(c, 0, clen); + chacha20_encrypt_bytes(&ctx, c, c, clen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +static int +stream_ref_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, uint64_t ic, + const unsigned char *k) +{ + struct chacha_ctx ctx; + uint8_t ic_bytes[8]; + uint32_t ic_high; + uint32_t ic_low; + + if (!mlen) { + return 0; + } + ic_high = U32V(ic >> 32); + ic_low = U32V(ic); + STORE32_LE(&ic_bytes[0], ic_low); + STORE32_LE(&ic_bytes[4], ic_high); + chacha_keysetup(&ctx, k); + chacha_ivsetup(&ctx, n, ic_bytes); + chacha20_encrypt_bytes(&ctx, m, c, mlen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +static int +stream_ietf_ext_ref_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + uint32_t ic, const unsigned char *k) +{ + struct chacha_ctx ctx; + uint8_t ic_bytes[4]; + + if (!mlen) { + return 0; + } + STORE32_LE(ic_bytes, ic); + chacha_keysetup(&ctx, k); + chacha_ietf_ivsetup(&ctx, n, ic_bytes); + chacha20_encrypt_bytes(&ctx, m, c, mlen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +struct crypto_stream_chacha20_implementation + crypto_stream_chacha20_ref_implementation = { + SODIUM_C99(.stream =) stream_ref, + SODIUM_C99(.stream_ietf_ext =) stream_ietf_ext_ref, + SODIUM_C99(.stream_xor_ic =) stream_ref_xor_ic, + SODIUM_C99(.stream_ietf_ext_xor_ic =) stream_ietf_ext_ref_xor_ic + }; diff --git a/sodium/sodium_chacha20-ref.h b/sodium/sodium_chacha20-ref.h new file mode 100644 index 00000000..4df4c0f0 --- /dev/null +++ b/sodium/sodium_chacha20-ref.h @@ -0,0 +1,8 @@ + +#include + +#include "sodium_stream_chacha20.h" +#include "sodium_crypto_stream_chacha20.h" + +extern struct crypto_stream_chacha20_implementation + crypto_stream_chacha20_ref_implementation; diff --git a/sodium/sodium_codecs.c b/sodium/sodium_codecs.c new file mode 100644 index 00000000..527c8ccf --- /dev/null +++ b/sodium/sodium_codecs.c @@ -0,0 +1,333 @@ +#include +#include +#include +#include +#include +#include +#include + +#include "sodium_core.h" +#include "sodium_utils.h" + +/* Derived from original code by CodesInChaos */ +char * +sodium_bin2hex(char *const hex, const size_t hex_maxlen, + const unsigned char *const bin, const size_t bin_len) +{ + size_t i = (size_t) 0U; + unsigned int x; + int b; + int c; + + if (bin_len >= SIZE_MAX / 2 || hex_maxlen <= bin_len * 2U) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } + while (i < bin_len) { + c = bin[i] & 0xf; + b = bin[i] >> 4; + x = (unsigned char) (87U + c + (((c - 10U) >> 8) & ~38U)) << 8 | + (unsigned char) (87U + b + (((b - 10U) >> 8) & ~38U)); + hex[i * 2U] = (char) x; + x >>= 8; + hex[i * 2U + 1U] = (char) x; + i++; + } + hex[i * 2U] = 0U; + + return hex; +} + +int +sodium_hex2bin(unsigned char *const bin, const size_t bin_maxlen, + const char *const hex, const size_t hex_len, + const char *const ignore, size_t *const bin_len, + const char **const hex_end) +{ + size_t bin_pos = (size_t) 0U; + size_t hex_pos = (size_t) 0U; + int ret = 0; + unsigned char c; + unsigned char c_acc = 0U; + unsigned char c_alpha0, c_alpha; + unsigned char c_num0, c_num; + unsigned char c_val; + unsigned char state = 0U; + + while (hex_pos < hex_len) { + c = (unsigned char) hex[hex_pos]; + c_num = c ^ 48U; + c_num0 = (c_num - 10U) >> 8; + c_alpha = (c & ~32U) - 55U; + c_alpha0 = ((c_alpha - 10U) ^ (c_alpha - 16U)) >> 8; + if ((c_num0 | c_alpha0) == 0U) { + if (ignore != NULL && state == 0U && strchr(ignore, c) != NULL) { + hex_pos++; + continue; + } + break; + } + c_val = (c_num0 & c_num) | (c_alpha0 & c_alpha); + if (bin_pos >= bin_maxlen) { + ret = -1; + errno = ERANGE; + break; + } + if (state == 0U) { + c_acc = c_val * 16U; + } else { + bin[bin_pos++] = c_acc | c_val; + } + state = ~state; + hex_pos++; + } + if (state != 0U) { + hex_pos--; + errno = EINVAL; + ret = -1; + } + if (ret != 0) { + bin_pos = (size_t) 0U; + } + if (hex_end != NULL) { + *hex_end = &hex[hex_pos]; + } else if (hex_pos != hex_len) { + errno = EINVAL; + ret = -1; + } + if (bin_len != NULL) { + *bin_len = bin_pos; + } + return ret; +} + +/* + * Some macros for constant-time comparisons. These work over values in + * the 0..255 range. Returned value is 0x00 on "false", 0xFF on "true". + * + * Original code by Thomas Pornin. + */ +#define EQ(x, y) \ + ((((0U - ((unsigned int) (x) ^ (unsigned int) (y))) >> 8) & 0xFF) ^ 0xFF) +#define GT(x, y) ((((unsigned int) (y) - (unsigned int) (x)) >> 8) & 0xFF) +#define GE(x, y) (GT(y, x) ^ 0xFF) +#define LT(x, y) GT(y, x) +#define LE(x, y) GE(y, x) + +static int +b64_byte_to_char(unsigned int x) +{ + return (LT(x, 26) & (x + 'A')) | + (GE(x, 26) & LT(x, 52) & (x + ('a' - 26))) | + (GE(x, 52) & LT(x, 62) & (x + ('0' - 52))) | (EQ(x, 62) & '+') | + (EQ(x, 63) & '/'); +} + +static unsigned int +b64_char_to_byte(int c) +{ + const unsigned int x = + (GE(c, 'A') & LE(c, 'Z') & (c - 'A')) | + (GE(c, 'a') & LE(c, 'z') & (c - ('a' - 26))) | + (GE(c, '0') & LE(c, '9') & (c - ('0' - 52))) | (EQ(c, '+') & 62) | + (EQ(c, '/') & 63); + + return x | (EQ(x, 0) & (EQ(c, 'A') ^ 0xFF)); +} + +static int +b64_byte_to_urlsafe_char(unsigned int x) +{ + return (LT(x, 26) & (x + 'A')) | + (GE(x, 26) & LT(x, 52) & (x + ('a' - 26))) | + (GE(x, 52) & LT(x, 62) & (x + ('0' - 52))) | (EQ(x, 62) & '-') | + (EQ(x, 63) & '_'); +} + +static unsigned int +b64_urlsafe_char_to_byte(int c) +{ + const unsigned x = + (GE(c, 'A') & LE(c, 'Z') & (c - 'A')) | + (GE(c, 'a') & LE(c, 'z') & (c - ('a' - 26))) | + (GE(c, '0') & LE(c, '9') & (c - ('0' - 52))) | (EQ(c, '-') & 62) | + (EQ(c, '_') & 63); + + return x | (EQ(x, 0) & (EQ(c, 'A') ^ 0xFF)); +} + + +#define VARIANT_NO_PADDING_MASK 0x2U +#define VARIANT_URLSAFE_MASK 0x4U + +static void +sodium_base64_check_variant(const int variant) +{ + if ((((unsigned int) variant) & ~ 0x6U) != 0x1U) { + sodium_misuse(); + } +} + +size_t +sodium_base64_encoded_len(const size_t bin_len, const int variant) +{ + sodium_base64_check_variant(variant); + + return sodium_base64_ENCODED_LEN(bin_len, variant); +} + +char * +sodium_bin2base64(char * const b64, const size_t b64_maxlen, + const unsigned char * const bin, const size_t bin_len, + const int variant) +{ + size_t acc_len = (size_t) 0; + size_t b64_len; + size_t b64_pos = (size_t) 0; + size_t bin_pos = (size_t) 0; + size_t nibbles; + size_t remainder; + unsigned int acc = 0U; + + sodium_base64_check_variant(variant); + nibbles = bin_len / 3; + remainder = bin_len - 3 * nibbles; + b64_len = nibbles * 4; + if (remainder != 0) { + if ((((unsigned int) variant) & VARIANT_NO_PADDING_MASK) == 0U) { + b64_len += 4; + } else { + b64_len += 2 + (remainder >> 1); + } + } + if (b64_maxlen <= b64_len) { + sodium_misuse(); + } + if ((((unsigned int) variant) & VARIANT_URLSAFE_MASK) != 0U) { + while (bin_pos < bin_len) { + acc = (acc << 8) + bin[bin_pos++]; + acc_len += 8; + while (acc_len >= 6) { + acc_len -= 6; + b64[b64_pos++] = (char) b64_byte_to_urlsafe_char((acc >> acc_len) & 0x3F); + } + } + if (acc_len > 0) { + b64[b64_pos++] = (char) b64_byte_to_urlsafe_char((acc << (6 - acc_len)) & 0x3F); + } + } else { + while (bin_pos < bin_len) { + acc = (acc << 8) + bin[bin_pos++]; + acc_len += 8; + while (acc_len >= 6) { + acc_len -= 6; + b64[b64_pos++] = (char) b64_byte_to_char((acc >> acc_len) & 0x3F); + } + } + if (acc_len > 0) { + b64[b64_pos++] = (char) b64_byte_to_char((acc << (6 - acc_len)) & 0x3F); + } + } + assert(b64_pos <= b64_len); + while (b64_pos < b64_len) { + b64[b64_pos++] = '='; + } + do { + b64[b64_pos++] = 0U; + } while (b64_pos < b64_maxlen); + + return b64; +} + +static int +_sodium_base642bin_skip_padding(const char * const b64, const size_t b64_len, + size_t * const b64_pos_p, + const char * const ignore, size_t padding_len) +{ + int c; + + while (padding_len > 0) { + if (*b64_pos_p >= b64_len) { + errno = ERANGE; + return -1; + } + c = b64[*b64_pos_p]; + if (c == '=') { + padding_len--; + } else if (ignore == NULL || strchr(ignore, c) == NULL) { + errno = EINVAL; + return -1; + } + (*b64_pos_p)++; + } + return 0; +} + +int +sodium_base642bin(unsigned char * const bin, const size_t bin_maxlen, + const char * const b64, const size_t b64_len, + const char * const ignore, size_t * const bin_len, + const char ** const b64_end, const int variant) +{ + size_t acc_len = (size_t) 0; + size_t b64_pos = (size_t) 0; + size_t bin_pos = (size_t) 0; + int is_urlsafe; + int ret = 0; + unsigned int acc = 0U; + unsigned int d; + char c; + + sodium_base64_check_variant(variant); + is_urlsafe = ((unsigned int) variant) & VARIANT_URLSAFE_MASK; + while (b64_pos < b64_len) { + c = b64[b64_pos]; + if (is_urlsafe) { + d = b64_urlsafe_char_to_byte(c); + } else { + d = b64_char_to_byte(c); + } + if (d == 0xFF) { + if (ignore != NULL && strchr(ignore, c) != NULL) { + b64_pos++; + continue; + } + break; + } + acc = (acc << 6) + d; + acc_len += 6; + if (acc_len >= 8) { + acc_len -= 8; + if (bin_pos >= bin_maxlen) { + errno = ERANGE; + ret = -1; + break; + } + bin[bin_pos++] = (acc >> acc_len) & 0xFF; + } + b64_pos++; + } + if (acc_len > 4U || (acc & ((1U << acc_len) - 1U)) != 0U) { + ret = -1; + } else if (ret == 0 && + (((unsigned int) variant) & VARIANT_NO_PADDING_MASK) == 0U) { + ret = _sodium_base642bin_skip_padding(b64, b64_len, &b64_pos, ignore, + acc_len / 2); + } + if (ret != 0) { + bin_pos = (size_t) 0U; + } else if (ignore != NULL) { + while (b64_pos < b64_len && strchr(ignore, b64[b64_pos]) != NULL) { + b64_pos++; + } + } + if (b64_end != NULL) { + *b64_end = &b64[b64_pos]; + } else if (b64_pos != b64_len) { + errno = EINVAL; + ret = -1; + } + if (bin_len != NULL) { + *bin_len = bin_pos; + } + return ret; +} diff --git a/sodium/sodium_core.c b/sodium/sodium_core.c new file mode 100644 index 00000000..8c432bf0 --- /dev/null +++ b/sodium/sodium_core.c @@ -0,0 +1,218 @@ + +#include +#include +#include +#include +#ifdef _WIN32 +# include +#elif defined(HAVE_PTHREAD) +# include +#endif + +#include "sodium_core.h" +#include "sodium_crypto_generichash.h" +#include "sodium_crypto_onetimeauth.h" +#include "sodium_crypto_scalarmult.h" +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_crypto_stream_salsa20.h" +#include "sodium_randombytes.h" +#include "sodium_runtime.h" +#include "sodium_utils.h" +#include "sodium_private_implementations.h" +#include "sodium_private_mutex.h" + +static volatile int initialized; +#ifdef _WIN32 +static volatile int locked; +#endif // #ifdef _WIN32 + +int +sodium_init(void) +{ + if (sodium_crit_enter() != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + if (initialized != 0) { + if (sodium_crit_leave() != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + return 1; + } + _sodium_runtime_get_cpu_features(); + randombytes_stir(); + _sodium_alloc_init(); + _crypto_generichash_blake2b_pick_best_implementation(); + _crypto_onetimeauth_poly1305_pick_best_implementation(); + _crypto_scalarmult_curve25519_pick_best_implementation(); + _crypto_stream_chacha20_pick_best_implementation(); + _crypto_stream_salsa20_pick_best_implementation(); + initialized = 1; + if (sodium_crit_leave() != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + #if NETCODE_CRYPTO_LOGS + printf( "\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + return 0; +} + +#ifdef _WIN32 + +static CRITICAL_SECTION _sodium_lock; +static volatile LONG _sodium_lock_initialized; + +int +_sodium_crit_init(void) +{ + LONG status = 0L; + + while ((status = InterlockedCompareExchange(&_sodium_lock_initialized, + 1L, 0L)) == 1L) { + Sleep(0); + } + + switch (status) { + case 0L: + InitializeCriticalSection(&_sodium_lock); + return InterlockedExchange(&_sodium_lock_initialized, 2L) == 1L ? 0 : -1; + case 2L: + return 0; + default: /* should never be reached */ + return -1; + } +} + +int +sodium_crit_enter(void) +{ + if (_sodium_crit_init() != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + EnterCriticalSection(&_sodium_lock); + assert(locked == 0); + locked = 1; + + return 0; +} + +int +sodium_crit_leave(void) +{ + if (locked == 0) { +# ifdef EPERM + errno = EPERM; +# endif + return -1; + } + locked = 0; + LeaveCriticalSection(&_sodium_lock); + + return 0; +} + +#elif defined(HAVE_PTHREAD) && !defined(__EMSCRIPTEN__) + +static pthread_mutex_t _sodium_lock = PTHREAD_MUTEX_INITIALIZER; + +int +sodium_crit_enter(void) +{ + int ret; + + if ((ret = pthread_mutex_lock(&_sodium_lock)) == 0) { + assert(locked == 0); + locked = 1; + } + return ret; +} + +int +sodium_crit_leave(void) +{ + if (locked == 0) { +# ifdef EPERM + errno = EPERM; +# endif + return -1; + } + locked = 0; + + return pthread_mutex_unlock(&_sodium_lock); +} + +#elif defined(HAVE_ATOMIC_OPS) && !defined(__EMSCRIPTEN__) && !defined(__native_client__) + +static volatile int _sodium_lock; + +int +sodium_crit_enter(void) +{ +# ifdef HAVE_NANOSLEEP + struct timespec q; + memset(&q, 0, sizeof q); +# endif + while (__sync_lock_test_and_set(&_sodium_lock, 1) != 0) { +# ifdef HAVE_NANOSLEEP + (void) nanosleep(&q, NULL); +# elif defined(__x86_64__) || defined(__i386__) + __asm__ __volatile__ ("pause"); +# endif + } + return 0; +} + +int +sodium_crit_leave(void) +{ + __sync_lock_release(&_sodium_lock); + + return 0; +} + +#else + +int +sodium_crit_enter(void) +{ + return 0; +} + +int +sodium_crit_leave(void) +{ + return 0; +} + +#endif + +static void (*_misuse_handler)(void); + +void +sodium_misuse(void) +{ + void (*handler)(void); + + (void) sodium_crit_leave(); + if (sodium_crit_enter() == 0) { + handler = _misuse_handler; + if (handler != NULL) { + handler(); + } + } +/* LCOV_EXCL_START */ + abort(); +} +/* LCOV_EXCL_STOP */ + +int +sodium_set_misuse_handler(void (*handler)(void)) +{ + if (sodium_crit_enter() != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + _misuse_handler = handler; + if (sodium_crit_leave() != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + return 0; +} diff --git a/windows/sodium/core.h b/sodium/sodium_core.h similarity index 55% rename from windows/sodium/core.h rename to sodium/sodium_core.h index 3ca44762..29481a0d 100644 --- a/windows/sodium/core.h +++ b/sodium/sodium_core.h @@ -2,16 +2,22 @@ #ifndef sodium_core_H #define sodium_core_H -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif -SODIUM_EXPORT int sodium_init(void) __attribute__ ((warn_unused_result)); +/* ---- */ + +int sodium_set_misuse_handler(void (*handler)(void)); + +void sodium_misuse(void) + __attribute__ ((noreturn)); + #ifdef __cplusplus } #endif diff --git a/sodium/sodium_core_ed25519.c b/sodium/sodium_core_ed25519.c new file mode 100644 index 00000000..c8a0d037 --- /dev/null +++ b/sodium/sodium_core_ed25519.c @@ -0,0 +1,195 @@ + +#include + +#include "sodium_crypto_core_ed25519.h" +#include "sodium_private_common.h" +#include "sodium_private_ed25519_ref10.h" +#include "sodium_randombytes.h" +#include "sodium_utils.h" + +int +crypto_core_ed25519_is_valid_point(const unsigned char *p) +{ + ge25519_p3 p_p3; + + if (ge25519_is_canonical(p) == 0 || + ge25519_has_small_order(p) != 0 || + ge25519_frombytes(&p_p3, p) != 0 || + ge25519_is_on_curve(&p_p3) == 0 || + ge25519_is_on_main_subgroup(&p_p3) == 0) { + return 0; + } + return 1; +} + +int +crypto_core_ed25519_add(unsigned char *r, + const unsigned char *p, const unsigned char *q) +{ + ge25519_p3 p_p3, q_p3, r_p3; + ge25519_p1p1 r_p1p1; + ge25519_cached q_cached; + + if (ge25519_frombytes(&p_p3, p) != 0 || ge25519_is_on_curve(&p_p3) == 0 || + ge25519_frombytes(&q_p3, q) != 0 || ge25519_is_on_curve(&q_p3) == 0) { + return -1; + } + ge25519_p3_to_cached(&q_cached, &q_p3); + ge25519_add(&r_p1p1, &p_p3, &q_cached); + ge25519_p1p1_to_p3(&r_p3, &r_p1p1); + ge25519_p3_tobytes(r, &r_p3); + + return 0; +} + +int +crypto_core_ed25519_sub(unsigned char *r, + const unsigned char *p, const unsigned char *q) +{ + ge25519_p3 p_p3, q_p3, r_p3; + ge25519_p1p1 r_p1p1; + ge25519_cached q_cached; + + if (ge25519_frombytes(&p_p3, p) != 0 || ge25519_is_on_curve(&p_p3) == 0 || + ge25519_frombytes(&q_p3, q) != 0 || ge25519_is_on_curve(&q_p3) == 0) { + return -1; + } + ge25519_p3_to_cached(&q_cached, &q_p3); + ge25519_sub(&r_p1p1, &p_p3, &q_cached); + ge25519_p1p1_to_p3(&r_p3, &r_p1p1); + ge25519_p3_tobytes(r, &r_p3); + + return 0; +} + +int +crypto_core_ed25519_from_uniform(unsigned char *p, const unsigned char *r) +{ + ge25519_from_uniform(p, r); + + return - ge25519_has_small_order(p); +} + +void +crypto_core_ed25519_scalar_random(unsigned char *r) +{ + do { + randombytes_buf(r, crypto_core_ed25519_SCALARBYTES); + r[crypto_core_ed25519_SCALARBYTES - 1] &= 0x1f; + } while (sc25519_is_canonical(r) == 0 || + sodium_is_zero(r, crypto_core_ed25519_SCALARBYTES)); +} + +int +crypto_core_ed25519_scalar_invert(unsigned char *recip, const unsigned char *s) +{ + sc25519_invert(recip, s); + + return - sodium_is_zero(s, crypto_core_ed25519_SCALARBYTES); +} + +/* 2^252+27742317777372353535851937790883648493 */ +static const unsigned char L[] = { + 0xed, 0xd3, 0xf5, 0x5c, 0x1a, 0x63, 0x12, 0x58, 0xd6, 0x9c, 0xf7, + 0xa2, 0xde, 0xf9, 0xde, 0x14, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10 +}; + +void +crypto_core_ed25519_scalar_negate(unsigned char *neg, const unsigned char *s) +{ + unsigned char t_[crypto_core_ed25519_NONREDUCEDSCALARBYTES]; + unsigned char s_[crypto_core_ed25519_NONREDUCEDSCALARBYTES]; + + COMPILER_ASSERT(crypto_core_ed25519_NONREDUCEDSCALARBYTES >= + 2 * crypto_core_ed25519_SCALARBYTES); + memset(t_, 0, sizeof t_); + memset(s_, 0, sizeof s_); + memcpy(t_ + crypto_core_ed25519_SCALARBYTES, L, + crypto_core_ed25519_SCALARBYTES); + memcpy(s_, s, crypto_core_ed25519_SCALARBYTES); + sodium_sub(t_, s_, sizeof t_); + sc25519_reduce(t_); + memcpy(neg, t_, crypto_core_ed25519_SCALARBYTES); +} + +void +crypto_core_ed25519_scalar_complement(unsigned char *comp, + const unsigned char *s) +{ + unsigned char t_[crypto_core_ed25519_NONREDUCEDSCALARBYTES]; + unsigned char s_[crypto_core_ed25519_NONREDUCEDSCALARBYTES]; + + COMPILER_ASSERT(crypto_core_ed25519_NONREDUCEDSCALARBYTES >= + 2 * crypto_core_ed25519_SCALARBYTES); + memset(t_, 0, sizeof t_); + memset(s_, 0, sizeof s_); + t_[0]++; + memcpy(t_ + crypto_core_ed25519_SCALARBYTES, L, + crypto_core_ed25519_SCALARBYTES); + memcpy(s_, s, crypto_core_ed25519_SCALARBYTES); + sodium_sub(t_, s_, sizeof t_); + sc25519_reduce(t_); + memcpy(comp, t_, crypto_core_ed25519_SCALARBYTES); +} + +void +crypto_core_ed25519_scalar_add(unsigned char *z, const unsigned char *x, + const unsigned char *y) +{ + unsigned char x_[crypto_core_ed25519_NONREDUCEDSCALARBYTES]; + unsigned char y_[crypto_core_ed25519_NONREDUCEDSCALARBYTES]; + + memset(x_, 0, sizeof x_); + memset(y_, 0, sizeof y_); + memcpy(x_, x, crypto_core_ed25519_SCALARBYTES); + memcpy(y_, y, crypto_core_ed25519_SCALARBYTES); + sodium_add(x_, y_, crypto_core_ed25519_SCALARBYTES); + crypto_core_ed25519_scalar_reduce(z, x_); +} + +void +crypto_core_ed25519_scalar_sub(unsigned char *z, const unsigned char *x, + const unsigned char *y) +{ + unsigned char yn[crypto_core_ed25519_SCALARBYTES]; + + crypto_core_ed25519_scalar_negate(yn, y); + crypto_core_ed25519_scalar_add(z, x, yn); +} + +void +crypto_core_ed25519_scalar_reduce(unsigned char *r, + const unsigned char *s) +{ + unsigned char t[crypto_core_ed25519_NONREDUCEDSCALARBYTES]; + + memcpy(t, s, sizeof t); + sc25519_reduce(t); + memcpy(r, t, crypto_core_ed25519_SCALARBYTES); + sodium_memzero(t, sizeof t); +} + +size_t +crypto_core_ed25519_bytes(void) +{ + return crypto_core_ed25519_BYTES; +} + +size_t +crypto_core_ed25519_nonreducedscalarbytes(void) +{ + return crypto_core_ed25519_NONREDUCEDSCALARBYTES; +} + +size_t +crypto_core_ed25519_uniformbytes(void) +{ + return crypto_core_ed25519_UNIFORMBYTES; +} + +size_t +crypto_core_ed25519_scalarbytes(void) +{ + return crypto_core_ed25519_SCALARBYTES; +} diff --git a/sodium/sodium_core_hchacha20.c b/sodium/sodium_core_hchacha20.c new file mode 100644 index 00000000..1c2437b2 --- /dev/null +++ b/sodium/sodium_core_hchacha20.c @@ -0,0 +1,93 @@ + +#include +#include + +#include "sodium_crypto_core_hchacha20.h" +#include "sodium_private_common.h" + +#define QUARTERROUND(A, B, C, D) \ + do { \ + A += B; D = ROTL32(D ^ A, 16); \ + C += D; B = ROTL32(B ^ C, 12); \ + A += B; D = ROTL32(D ^ A, 8); \ + C += D; B = ROTL32(B ^ C, 7); \ + } while(0) + +int +crypto_core_hchacha20(unsigned char *out, const unsigned char *in, + const unsigned char *k, const unsigned char *c) +{ + int i; + uint32_t x0, x1, x2, x3, x4, x5, x6, x7; + uint32_t x8, x9, x10, x11, x12, x13, x14, x15; + + if (c == NULL) { + x0 = 0x61707865; + x1 = 0x3320646e; + x2 = 0x79622d32; + x3 = 0x6b206574; + } else { + x0 = LOAD32_LE(c + 0); + x1 = LOAD32_LE(c + 4); + x2 = LOAD32_LE(c + 8); + x3 = LOAD32_LE(c + 12); + } + x4 = LOAD32_LE(k + 0); + x5 = LOAD32_LE(k + 4); + x6 = LOAD32_LE(k + 8); + x7 = LOAD32_LE(k + 12); + x8 = LOAD32_LE(k + 16); + x9 = LOAD32_LE(k + 20); + x10 = LOAD32_LE(k + 24); + x11 = LOAD32_LE(k + 28); + x12 = LOAD32_LE(in + 0); + x13 = LOAD32_LE(in + 4); + x14 = LOAD32_LE(in + 8); + x15 = LOAD32_LE(in + 12); + + for (i = 0; i < 10; i++) { + QUARTERROUND(x0, x4, x8, x12); + QUARTERROUND(x1, x5, x9, x13); + QUARTERROUND(x2, x6, x10, x14); + QUARTERROUND(x3, x7, x11, x15); + QUARTERROUND(x0, x5, x10, x15); + QUARTERROUND(x1, x6, x11, x12); + QUARTERROUND(x2, x7, x8, x13); + QUARTERROUND(x3, x4, x9, x14); + } + + STORE32_LE(out + 0, x0); + STORE32_LE(out + 4, x1); + STORE32_LE(out + 8, x2); + STORE32_LE(out + 12, x3); + STORE32_LE(out + 16, x12); + STORE32_LE(out + 20, x13); + STORE32_LE(out + 24, x14); + STORE32_LE(out + 28, x15); + + return 0; +} + +size_t +crypto_core_hchacha20_outputbytes(void) +{ + return crypto_core_hchacha20_OUTPUTBYTES; +} + +size_t +crypto_core_hchacha20_inputbytes(void) +{ + return crypto_core_hchacha20_INPUTBYTES; +} + +size_t +crypto_core_hchacha20_keybytes(void) +{ + return crypto_core_hchacha20_KEYBYTES; +} + +size_t +crypto_core_hchacha20_constbytes(void) +{ + return crypto_core_hchacha20_CONSTBYTES; +} diff --git a/sodium/sodium_core_hsalsa20.c b/sodium/sodium_core_hsalsa20.c new file mode 100644 index 00000000..3050ac1e --- /dev/null +++ b/sodium/sodium_core_hsalsa20.c @@ -0,0 +1,21 @@ +#include "sodium_crypto_core_hsalsa20.h" + +size_t +crypto_core_hsalsa20_outputbytes(void) { + return crypto_core_hsalsa20_OUTPUTBYTES; +} + +size_t +crypto_core_hsalsa20_inputbytes(void) { + return crypto_core_hsalsa20_INPUTBYTES; +} + +size_t +crypto_core_hsalsa20_keybytes(void) { + return crypto_core_hsalsa20_KEYBYTES; +} + +size_t +crypto_core_hsalsa20_constbytes(void) { + return crypto_core_hsalsa20_CONSTBYTES; +} diff --git a/sodium/sodium_core_hsalsa20_ref2.c b/sodium/sodium_core_hsalsa20_ref2.c new file mode 100644 index 00000000..52523ec9 --- /dev/null +++ b/sodium/sodium_core_hsalsa20_ref2.c @@ -0,0 +1,95 @@ +/* +version 20080912 +D. J. Bernstein +Public domain. +*/ + +#include +#include + +#include "sodium_crypto_core_hsalsa20.h" +#include "sodium_private_common.h" + +#define ROUNDS 20 +#define U32C(v) (v##U) + +int +crypto_core_hsalsa20(unsigned char *out, + const unsigned char *in, + const unsigned char *k, + const unsigned char *c) +{ + uint32_t x0, x1, x2, x3, x4, x5, x6, x7, x8, + x9, x10, x11, x12, x13, x14, x15; + int i; + + if (c == NULL) { + x0 = U32C(0x61707865); + x5 = U32C(0x3320646e); + x10 = U32C(0x79622d32); + x15 = U32C(0x6b206574); + } else { + x0 = LOAD32_LE(c + 0); + x5 = LOAD32_LE(c + 4); + x10 = LOAD32_LE(c + 8); + x15 = LOAD32_LE(c + 12); + } + x1 = LOAD32_LE(k + 0); + x2 = LOAD32_LE(k + 4); + x3 = LOAD32_LE(k + 8); + x4 = LOAD32_LE(k + 12); + x11 = LOAD32_LE(k + 16); + x12 = LOAD32_LE(k + 20); + x13 = LOAD32_LE(k + 24); + x14 = LOAD32_LE(k + 28); + x6 = LOAD32_LE(in + 0); + x7 = LOAD32_LE(in + 4); + x8 = LOAD32_LE(in + 8); + x9 = LOAD32_LE(in + 12); + + for (i = ROUNDS; i > 0; i -= 2) { + x4 ^= ROTL32(x0 + x12, 7); + x8 ^= ROTL32(x4 + x0, 9); + x12 ^= ROTL32(x8 + x4, 13); + x0 ^= ROTL32(x12 + x8, 18); + x9 ^= ROTL32(x5 + x1, 7); + x13 ^= ROTL32(x9 + x5, 9); + x1 ^= ROTL32(x13 + x9, 13); + x5 ^= ROTL32(x1 + x13, 18); + x14 ^= ROTL32(x10 + x6, 7); + x2 ^= ROTL32(x14 + x10, 9); + x6 ^= ROTL32(x2 + x14, 13); + x10 ^= ROTL32(x6 + x2, 18); + x3 ^= ROTL32(x15 + x11, 7); + x7 ^= ROTL32(x3 + x15, 9); + x11 ^= ROTL32(x7 + x3, 13); + x15 ^= ROTL32(x11 + x7, 18); + x1 ^= ROTL32(x0 + x3, 7); + x2 ^= ROTL32(x1 + x0, 9); + x3 ^= ROTL32(x2 + x1, 13); + x0 ^= ROTL32(x3 + x2, 18); + x6 ^= ROTL32(x5 + x4, 7); + x7 ^= ROTL32(x6 + x5, 9); + x4 ^= ROTL32(x7 + x6, 13); + x5 ^= ROTL32(x4 + x7, 18); + x11 ^= ROTL32(x10 + x9, 7); + x8 ^= ROTL32(x11 + x10, 9); + x9 ^= ROTL32(x8 + x11, 13); + x10 ^= ROTL32(x9 + x8, 18); + x12 ^= ROTL32(x15 + x14, 7); + x13 ^= ROTL32(x12 + x15, 9); + x14 ^= ROTL32(x13 + x12, 13); + x15 ^= ROTL32(x14 + x13, 18); + } + + STORE32_LE(out + 0, x0); + STORE32_LE(out + 4, x5); + STORE32_LE(out + 8, x10); + STORE32_LE(out + 12, x15); + STORE32_LE(out + 16, x6); + STORE32_LE(out + 20, x7); + STORE32_LE(out + 24, x8); + STORE32_LE(out + 28, x9); + + return 0; +} diff --git a/sodium/sodium_core_salsa_ref.c b/sodium/sodium_core_salsa_ref.c new file mode 100644 index 00000000..be0b4108 --- /dev/null +++ b/sodium/sodium_core_salsa_ref.c @@ -0,0 +1,195 @@ + +#include +#include + +#include "sodium_crypto_core_salsa20.h" +#include "sodium_crypto_core_salsa2012.h" +#include "sodium_crypto_core_salsa208.h" +#include "sodium_private_common.h" + +static void +crypto_core_salsa(unsigned char *out, const unsigned char *in, + const unsigned char *k, const unsigned char *c, + const int rounds) +{ + uint32_t x0, x1, x2, x3, x4, x5, x6, x7, x8, x9, x10, x11, x12, x13, x14, + x15; + uint32_t j0, j1, j2, j3, j4, j5, j6, j7, j8, j9, j10, j11, j12, j13, j14, + j15; + int i; + + j0 = x0 = 0x61707865; + j5 = x5 = 0x3320646e; + j10 = x10 = 0x79622d32; + j15 = x15 = 0x6b206574; + if (c != NULL) { + j0 = x0 = LOAD32_LE(c + 0); + j5 = x5 = LOAD32_LE(c + 4); + j10 = x10 = LOAD32_LE(c + 8); + j15 = x15 = LOAD32_LE(c + 12); + } + j1 = x1 = LOAD32_LE(k + 0); + j2 = x2 = LOAD32_LE(k + 4); + j3 = x3 = LOAD32_LE(k + 8); + j4 = x4 = LOAD32_LE(k + 12); + j11 = x11 = LOAD32_LE(k + 16); + j12 = x12 = LOAD32_LE(k + 20); + j13 = x13 = LOAD32_LE(k + 24); + j14 = x14 = LOAD32_LE(k + 28); + + j6 = x6 = LOAD32_LE(in + 0); + j7 = x7 = LOAD32_LE(in + 4); + j8 = x8 = LOAD32_LE(in + 8); + j9 = x9 = LOAD32_LE(in + 12); + + for (i = 0; i < rounds; i += 2) { + x4 ^= ROTL32(x0 + x12, 7); + x8 ^= ROTL32(x4 + x0, 9); + x12 ^= ROTL32(x8 + x4, 13); + x0 ^= ROTL32(x12 + x8, 18); + x9 ^= ROTL32(x5 + x1, 7); + x13 ^= ROTL32(x9 + x5, 9); + x1 ^= ROTL32(x13 + x9, 13); + x5 ^= ROTL32(x1 + x13, 18); + x14 ^= ROTL32(x10 + x6, 7); + x2 ^= ROTL32(x14 + x10, 9); + x6 ^= ROTL32(x2 + x14, 13); + x10 ^= ROTL32(x6 + x2, 18); + x3 ^= ROTL32(x15 + x11, 7); + x7 ^= ROTL32(x3 + x15, 9); + x11 ^= ROTL32(x7 + x3, 13); + x15 ^= ROTL32(x11 + x7, 18); + x1 ^= ROTL32(x0 + x3, 7); + x2 ^= ROTL32(x1 + x0, 9); + x3 ^= ROTL32(x2 + x1, 13); + x0 ^= ROTL32(x3 + x2, 18); + x6 ^= ROTL32(x5 + x4, 7); + x7 ^= ROTL32(x6 + x5, 9); + x4 ^= ROTL32(x7 + x6, 13); + x5 ^= ROTL32(x4 + x7, 18); + x11 ^= ROTL32(x10 + x9, 7); + x8 ^= ROTL32(x11 + x10, 9); + x9 ^= ROTL32(x8 + x11, 13); + x10 ^= ROTL32(x9 + x8, 18); + x12 ^= ROTL32(x15 + x14, 7); + x13 ^= ROTL32(x12 + x15, 9); + x14 ^= ROTL32(x13 + x12, 13); + x15 ^= ROTL32(x14 + x13, 18); + } + STORE32_LE(out + 0, x0 + j0); + STORE32_LE(out + 4, x1 + j1); + STORE32_LE(out + 8, x2 + j2); + STORE32_LE(out + 12, x3 + j3); + STORE32_LE(out + 16, x4 + j4); + STORE32_LE(out + 20, x5 + j5); + STORE32_LE(out + 24, x6 + j6); + STORE32_LE(out + 28, x7 + j7); + STORE32_LE(out + 32, x8 + j8); + STORE32_LE(out + 36, x9 + j9); + STORE32_LE(out + 40, x10 + j10); + STORE32_LE(out + 44, x11 + j11); + STORE32_LE(out + 48, x12 + j12); + STORE32_LE(out + 52, x13 + j13); + STORE32_LE(out + 56, x14 + j14); + STORE32_LE(out + 60, x15 + j15); +} + +int +crypto_core_salsa20(unsigned char *out, const unsigned char *in, + const unsigned char *k, const unsigned char *c) +{ + crypto_core_salsa(out, in, k, c, 20); + return 0; +} + +size_t +crypto_core_salsa20_outputbytes(void) +{ + return crypto_core_salsa20_OUTPUTBYTES; +} + +size_t +crypto_core_salsa20_inputbytes(void) +{ + return crypto_core_salsa20_INPUTBYTES; +} + +size_t +crypto_core_salsa20_keybytes(void) +{ + return crypto_core_salsa20_KEYBYTES; +} + +size_t +crypto_core_salsa20_constbytes(void) +{ + return crypto_core_salsa20_CONSTBYTES; +} + +#ifndef MINIMAL + +int +crypto_core_salsa2012(unsigned char *out, const unsigned char *in, + const unsigned char *k, const unsigned char *c) +{ + crypto_core_salsa(out, in, k, c, 12); + return 0; +} + +size_t +crypto_core_salsa2012_outputbytes(void) +{ + return crypto_core_salsa2012_OUTPUTBYTES; +} + +size_t +crypto_core_salsa2012_inputbytes(void) +{ + return crypto_core_salsa2012_INPUTBYTES; +} + +size_t +crypto_core_salsa2012_keybytes(void) +{ + return crypto_core_salsa2012_KEYBYTES; +} + +size_t +crypto_core_salsa2012_constbytes(void) +{ + return crypto_core_salsa2012_CONSTBYTES; +} + +int +crypto_core_salsa208(unsigned char *out, const unsigned char *in, + const unsigned char *k, const unsigned char *c) +{ + crypto_core_salsa(out, in, k, c, 8); + return 0; +} + +size_t +crypto_core_salsa208_outputbytes(void) +{ + return crypto_core_salsa208_OUTPUTBYTES; +} + +size_t +crypto_core_salsa208_inputbytes(void) +{ + return crypto_core_salsa208_INPUTBYTES; +} + +size_t +crypto_core_salsa208_keybytes(void) +{ + return crypto_core_salsa208_KEYBYTES; +} + +size_t +crypto_core_salsa208_constbytes(void) +{ + return crypto_core_salsa208_CONSTBYTES; +} + +#endif diff --git a/windows/sodium/crypto_aead_chacha20poly1305.h b/sodium/sodium_crypto_aead_chacha20poly1305.h similarity index 79% rename from windows/sodium/crypto_aead_chacha20poly1305.h rename to sodium/sodium_crypto_aead_chacha20poly1305.h index 0bbc6885..d037be8f 100644 --- a/windows/sodium/crypto_aead_chacha20poly1305.h +++ b/sodium/sodium_crypto_aead_chacha20poly1305.h @@ -2,7 +2,8 @@ #define crypto_aead_chacha20poly1305_H #include -#include "export.h" + +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -14,23 +15,23 @@ extern "C" { /* -- IETF ChaCha20-Poly1305 construction with a 96-bit nonce and a 32-bit internal counter -- */ #define crypto_aead_chacha20poly1305_ietf_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_aead_chacha20poly1305_ietf_keybytes(void); #define crypto_aead_chacha20poly1305_ietf_NSECBYTES 0U -SODIUM_EXPORT size_t crypto_aead_chacha20poly1305_ietf_nsecbytes(void); #define crypto_aead_chacha20poly1305_ietf_NPUBBYTES 12U -SODIUM_EXPORT size_t crypto_aead_chacha20poly1305_ietf_npubbytes(void); #define crypto_aead_chacha20poly1305_ietf_ABYTES 16U -SODIUM_EXPORT size_t crypto_aead_chacha20poly1305_ietf_abytes(void); -SODIUM_EXPORT +#define crypto_aead_chacha20poly1305_ietf_MESSAGEBYTES_MAX \ + SODIUM_MIN(SODIUM_SIZE_MAX - crypto_aead_chacha20poly1305_ietf_ABYTES, \ + (64ULL * ((1ULL << 32) - 1ULL))) +size_t crypto_aead_chacha20poly1305_ietf_messagebytes_max(void); + int crypto_aead_chacha20poly1305_ietf_encrypt(unsigned char *c, unsigned long long *clen_p, const unsigned char *m, @@ -39,9 +40,9 @@ int crypto_aead_chacha20poly1305_ietf_encrypt(unsigned char *c, unsigned long long adlen, const unsigned char *nsec, const unsigned char *npub, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull(1, 8, 9))); -SODIUM_EXPORT int crypto_aead_chacha20poly1305_ietf_decrypt(unsigned char *m, unsigned long long *mlen_p, unsigned char *nsec, @@ -51,9 +52,8 @@ int crypto_aead_chacha20poly1305_ietf_decrypt(unsigned char *m, unsigned long long adlen, const unsigned char *npub, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(4, 8, 9))); -SODIUM_EXPORT int crypto_aead_chacha20poly1305_ietf_encrypt_detached(unsigned char *c, unsigned char *mac, unsigned long long *maclen_p, @@ -63,9 +63,9 @@ int crypto_aead_chacha20poly1305_ietf_encrypt_detached(unsigned char *c, unsigned long long adlen, const unsigned char *nsec, const unsigned char *npub, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull(1, 2, 9, 10))); -SODIUM_EXPORT int crypto_aead_chacha20poly1305_ietf_decrypt_detached(unsigned char *m, unsigned char *nsec, const unsigned char *c, @@ -75,30 +75,29 @@ int crypto_aead_chacha20poly1305_ietf_decrypt_detached(unsigned char *m, unsigned long long adlen, const unsigned char *npub, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(3, 5, 8, 9))); -SODIUM_EXPORT -void crypto_aead_chacha20poly1305_ietf_keygen(unsigned char k[crypto_aead_chacha20poly1305_ietf_KEYBYTES]); +void crypto_aead_chacha20poly1305_ietf_keygen(unsigned char k[crypto_aead_chacha20poly1305_ietf_KEYBYTES]) + __attribute__ ((nonnull)); /* -- Original ChaCha20-Poly1305 construction with a 64-bit nonce and a 64-bit internal counter -- */ #define crypto_aead_chacha20poly1305_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_aead_chacha20poly1305_keybytes(void); #define crypto_aead_chacha20poly1305_NSECBYTES 0U -SODIUM_EXPORT size_t crypto_aead_chacha20poly1305_nsecbytes(void); #define crypto_aead_chacha20poly1305_NPUBBYTES 8U -SODIUM_EXPORT size_t crypto_aead_chacha20poly1305_npubbytes(void); #define crypto_aead_chacha20poly1305_ABYTES 16U -SODIUM_EXPORT size_t crypto_aead_chacha20poly1305_abytes(void); -SODIUM_EXPORT +#define crypto_aead_chacha20poly1305_MESSAGEBYTES_MAX \ + (SODIUM_SIZE_MAX - crypto_aead_chacha20poly1305_ABYTES) +size_t crypto_aead_chacha20poly1305_messagebytes_max(void); + int crypto_aead_chacha20poly1305_encrypt(unsigned char *c, unsigned long long *clen_p, const unsigned char *m, @@ -107,9 +106,9 @@ int crypto_aead_chacha20poly1305_encrypt(unsigned char *c, unsigned long long adlen, const unsigned char *nsec, const unsigned char *npub, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull(1, 8, 9))); -SODIUM_EXPORT int crypto_aead_chacha20poly1305_decrypt(unsigned char *m, unsigned long long *mlen_p, unsigned char *nsec, @@ -119,9 +118,8 @@ int crypto_aead_chacha20poly1305_decrypt(unsigned char *m, unsigned long long adlen, const unsigned char *npub, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(4, 8, 9))); -SODIUM_EXPORT int crypto_aead_chacha20poly1305_encrypt_detached(unsigned char *c, unsigned char *mac, unsigned long long *maclen_p, @@ -131,9 +129,9 @@ int crypto_aead_chacha20poly1305_encrypt_detached(unsigned char *c, unsigned long long adlen, const unsigned char *nsec, const unsigned char *npub, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull(1, 2, 9, 10))); -SODIUM_EXPORT int crypto_aead_chacha20poly1305_decrypt_detached(unsigned char *m, unsigned char *nsec, const unsigned char *c, @@ -143,17 +141,18 @@ int crypto_aead_chacha20poly1305_decrypt_detached(unsigned char *m, unsigned long long adlen, const unsigned char *npub, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(3, 5, 8, 9))); -SODIUM_EXPORT -void crypto_aead_chacha20poly1305_keygen(unsigned char k[crypto_aead_chacha20poly1305_KEYBYTES]); +void crypto_aead_chacha20poly1305_keygen(unsigned char k[crypto_aead_chacha20poly1305_KEYBYTES]) + __attribute__ ((nonnull)); /* Aliases */ -#define crypto_aead_chacha20poly1305_IETF_KEYBYTES crypto_aead_chacha20poly1305_ietf_KEYBYTES -#define crypto_aead_chacha20poly1305_IETF_NSECBYTES crypto_aead_chacha20poly1305_ietf_NSECBYTES -#define crypto_aead_chacha20poly1305_IETF_NPUBBYTES crypto_aead_chacha20poly1305_ietf_NPUBBYTES -#define crypto_aead_chacha20poly1305_IETF_ABYTES crypto_aead_chacha20poly1305_ietf_ABYTES +#define crypto_aead_chacha20poly1305_IETF_KEYBYTES crypto_aead_chacha20poly1305_ietf_KEYBYTES +#define crypto_aead_chacha20poly1305_IETF_NSECBYTES crypto_aead_chacha20poly1305_ietf_NSECBYTES +#define crypto_aead_chacha20poly1305_IETF_NPUBBYTES crypto_aead_chacha20poly1305_ietf_NPUBBYTES +#define crypto_aead_chacha20poly1305_IETF_ABYTES crypto_aead_chacha20poly1305_ietf_ABYTES +#define crypto_aead_chacha20poly1305_IETF_MESSAGEBYTES_MAX crypto_aead_chacha20poly1305_ietf_MESSAGEBYTES_MAX #ifdef __cplusplus } diff --git a/windows/sodium/crypto_aead_xchacha20poly1305.h b/sodium/sodium_crypto_aead_xchacha20poly1305.h similarity index 76% rename from windows/sodium/crypto_aead_xchacha20poly1305.h rename to sodium/sodium_crypto_aead_xchacha20poly1305.h index f863ce88..5a099a24 100644 --- a/windows/sodium/crypto_aead_xchacha20poly1305.h +++ b/sodium/sodium_crypto_aead_xchacha20poly1305.h @@ -2,7 +2,8 @@ #define crypto_aead_xchacha20poly1305_H #include -#include "export.h" + +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -12,22 +13,21 @@ extern "C" { #endif #define crypto_aead_xchacha20poly1305_ietf_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_aead_xchacha20poly1305_ietf_keybytes(void); #define crypto_aead_xchacha20poly1305_ietf_NSECBYTES 0U -SODIUM_EXPORT size_t crypto_aead_xchacha20poly1305_ietf_nsecbytes(void); #define crypto_aead_xchacha20poly1305_ietf_NPUBBYTES 24U -SODIUM_EXPORT size_t crypto_aead_xchacha20poly1305_ietf_npubbytes(void); #define crypto_aead_xchacha20poly1305_ietf_ABYTES 16U -SODIUM_EXPORT size_t crypto_aead_xchacha20poly1305_ietf_abytes(void); -SODIUM_EXPORT +#define crypto_aead_xchacha20poly1305_ietf_MESSAGEBYTES_MAX \ + (SODIUM_SIZE_MAX - crypto_aead_xchacha20poly1305_ietf_ABYTES) +size_t crypto_aead_xchacha20poly1305_ietf_messagebytes_max(void); + int crypto_aead_xchacha20poly1305_ietf_encrypt(unsigned char *c, unsigned long long *clen_p, const unsigned char *m, @@ -36,9 +36,9 @@ int crypto_aead_xchacha20poly1305_ietf_encrypt(unsigned char *c, unsigned long long adlen, const unsigned char *nsec, const unsigned char *npub, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull(1, 8, 9))); -SODIUM_EXPORT int crypto_aead_xchacha20poly1305_ietf_decrypt(unsigned char *m, unsigned long long *mlen_p, unsigned char *nsec, @@ -48,9 +48,8 @@ int crypto_aead_xchacha20poly1305_ietf_decrypt(unsigned char *m, unsigned long long adlen, const unsigned char *npub, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(4, 8, 9))); -SODIUM_EXPORT int crypto_aead_xchacha20poly1305_ietf_encrypt_detached(unsigned char *c, unsigned char *mac, unsigned long long *maclen_p, @@ -60,9 +59,9 @@ int crypto_aead_xchacha20poly1305_ietf_encrypt_detached(unsigned char *c, unsigned long long adlen, const unsigned char *nsec, const unsigned char *npub, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull(1, 2, 9, 10))); -SODIUM_EXPORT int crypto_aead_xchacha20poly1305_ietf_decrypt_detached(unsigned char *m, unsigned char *nsec, const unsigned char *c, @@ -72,17 +71,18 @@ int crypto_aead_xchacha20poly1305_ietf_decrypt_detached(unsigned char *m, unsigned long long adlen, const unsigned char *npub, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(3, 5, 9, 9))); -SODIUM_EXPORT -void crypto_aead_xchacha20poly1305_ietf_keygen(unsigned char k[crypto_aead_xchacha20poly1305_ietf_KEYBYTES]); +void crypto_aead_xchacha20poly1305_ietf_keygen(unsigned char k[crypto_aead_xchacha20poly1305_ietf_KEYBYTES]) + __attribute__ ((nonnull)); /* Aliases */ -#define crypto_aead_xchacha20poly1305_IETF_KEYBYTES crypto_aead_xchacha20poly1305_ietf_KEYBYTES -#define crypto_aead_xchacha20poly1305_IETF_NSECBYTES crypto_aead_xchacha20poly1305_ietf_NSECBYTES -#define crypto_aead_xchacha20poly1305_IETF_NPUBBYTES crypto_aead_xchacha20poly1305_ietf_NPUBBYTES -#define crypto_aead_xchacha20poly1305_IETF_ABYTES crypto_aead_xchacha20poly1305_ietf_ABYTES +#define crypto_aead_xchacha20poly1305_IETF_KEYBYTES crypto_aead_xchacha20poly1305_ietf_KEYBYTES +#define crypto_aead_xchacha20poly1305_IETF_NSECBYTES crypto_aead_xchacha20poly1305_ietf_NSECBYTES +#define crypto_aead_xchacha20poly1305_IETF_NPUBBYTES crypto_aead_xchacha20poly1305_ietf_NPUBBYTES +#define crypto_aead_xchacha20poly1305_IETF_ABYTES crypto_aead_xchacha20poly1305_ietf_ABYTES +#define crypto_aead_xchacha20poly1305_IETF_MESSAGEBYTES_MAX crypto_aead_xchacha20poly1305_ietf_MESSAGEBYTES_MAX #ifdef __cplusplus } diff --git a/windows/sodium/crypto_auth.h b/sodium/sodium_crypto_auth.h similarity index 72% rename from windows/sodium/crypto_auth.h rename to sodium/sodium_crypto_auth.h index 7174e7bc..b434a4f5 100644 --- a/windows/sodium/crypto_auth.h +++ b/sodium/sodium_crypto_auth.h @@ -3,8 +3,8 @@ #include -#include "crypto_auth_hmacsha512256.h" -#include "export.h" +#include "sodium_crypto_auth_hmacsha512256.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -14,28 +14,24 @@ extern "C" { #endif #define crypto_auth_BYTES crypto_auth_hmacsha512256_BYTES -SODIUM_EXPORT size_t crypto_auth_bytes(void); #define crypto_auth_KEYBYTES crypto_auth_hmacsha512256_KEYBYTES -SODIUM_EXPORT size_t crypto_auth_keybytes(void); #define crypto_auth_PRIMITIVE "hmacsha512256" -SODIUM_EXPORT const char *crypto_auth_primitive(void); -SODIUM_EXPORT int crypto_auth(unsigned char *out, const unsigned char *in, - unsigned long long inlen, const unsigned char *k); + unsigned long long inlen, const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_auth_verify(const unsigned char *h, const unsigned char *in, unsigned long long inlen, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_auth_keygen(unsigned char k[crypto_auth_KEYBYTES]); +void crypto_auth_keygen(unsigned char k[crypto_auth_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_auth_hmacsha512.h b/sodium/sodium_crypto_auth_hmacsha512.h similarity index 73% rename from windows/sodium/crypto_auth_hmacsha512.h rename to sodium/sodium_crypto_auth_hmacsha512.h index 77a55fbc..57f6d347 100644 --- a/windows/sodium/crypto_auth_hmacsha512.h +++ b/sodium/sodium_crypto_auth_hmacsha512.h @@ -2,8 +2,9 @@ #define crypto_auth_hmacsha512_H #include -#include "crypto_hash_sha512.h" -#include "export.h" + +#include "sodium_crypto_hash_sha512.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -13,25 +14,21 @@ extern "C" { #endif #define crypto_auth_hmacsha512_BYTES 64U -SODIUM_EXPORT size_t crypto_auth_hmacsha512_bytes(void); #define crypto_auth_hmacsha512_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_auth_hmacsha512_keybytes(void); -SODIUM_EXPORT int crypto_auth_hmacsha512(unsigned char *out, const unsigned char *in, unsigned long long inlen, - const unsigned char *k); + const unsigned char *k) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_auth_hmacsha512_verify(const unsigned char *h, const unsigned char *in, unsigned long long inlen, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); /* ------------------------------------------------------------------------- */ @@ -40,25 +37,21 @@ typedef struct crypto_auth_hmacsha512_state { crypto_hash_sha512_state octx; } crypto_auth_hmacsha512_state; -SODIUM_EXPORT size_t crypto_auth_hmacsha512_statebytes(void); -SODIUM_EXPORT int crypto_auth_hmacsha512_init(crypto_auth_hmacsha512_state *state, const unsigned char *key, - size_t keylen); + size_t keylen) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_auth_hmacsha512_update(crypto_auth_hmacsha512_state *state, const unsigned char *in, - unsigned long long inlen); + unsigned long long inlen) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_auth_hmacsha512_final(crypto_auth_hmacsha512_state *state, - unsigned char *out); + unsigned char *out) __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_auth_hmacsha512_keygen(unsigned char k[crypto_auth_hmacsha512_KEYBYTES]); +void crypto_auth_hmacsha512_keygen(unsigned char k[crypto_auth_hmacsha512_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_auth_hmacsha512256.h b/sodium/sodium_crypto_auth_hmacsha512256.h similarity index 75% rename from windows/sodium/crypto_auth_hmacsha512256.h rename to sodium/sodium_crypto_auth_hmacsha512256.h index 4842f3de..5c273c8b 100644 --- a/windows/sodium/crypto_auth_hmacsha512256.h +++ b/sodium/sodium_crypto_auth_hmacsha512256.h @@ -2,8 +2,9 @@ #define crypto_auth_hmacsha512256_H #include -#include "crypto_auth_hmacsha512.h" -#include "export.h" + +#include "sodium_crypto_auth_hmacsha512.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -13,47 +14,39 @@ extern "C" { #endif #define crypto_auth_hmacsha512256_BYTES 32U -SODIUM_EXPORT size_t crypto_auth_hmacsha512256_bytes(void); #define crypto_auth_hmacsha512256_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_auth_hmacsha512256_keybytes(void); -SODIUM_EXPORT int crypto_auth_hmacsha512256(unsigned char *out, const unsigned char *in, unsigned long long inlen,const unsigned char *k); -SODIUM_EXPORT int crypto_auth_hmacsha512256_verify(const unsigned char *h, const unsigned char *in, unsigned long long inlen, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); /* ------------------------------------------------------------------------- */ typedef crypto_auth_hmacsha512_state crypto_auth_hmacsha512256_state; -SODIUM_EXPORT size_t crypto_auth_hmacsha512256_statebytes(void); -SODIUM_EXPORT int crypto_auth_hmacsha512256_init(crypto_auth_hmacsha512256_state *state, const unsigned char *key, - size_t keylen); + size_t keylen) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_auth_hmacsha512256_update(crypto_auth_hmacsha512256_state *state, const unsigned char *in, - unsigned long long inlen); + unsigned long long inlen) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_auth_hmacsha512256_final(crypto_auth_hmacsha512256_state *state, - unsigned char *out); + unsigned char *out) __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_auth_hmacsha512256_keygen(unsigned char k[crypto_auth_hmacsha512256_KEYBYTES]); +void crypto_auth_hmacsha512256_keygen(unsigned char k[crypto_auth_hmacsha512256_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/sodium/sodium_crypto_box.c b/sodium/sodium_crypto_box.c new file mode 100644 index 00000000..2bdd3cc2 --- /dev/null +++ b/sodium/sodium_crypto_box.c @@ -0,0 +1,47 @@ + +#include "sodium_crypto_box.h" + +int +crypto_box_keypair(unsigned char *pk, unsigned char *sk) +{ + return crypto_box_curve25519xsalsa20poly1305_keypair(pk, sk); +} + +int +crypto_box_beforenm(unsigned char *k, const unsigned char *pk, + const unsigned char *sk) +{ + return crypto_box_curve25519xsalsa20poly1305_beforenm(k, pk, sk); +} + +int +crypto_box_afternm(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *k) +{ + return crypto_box_curve25519xsalsa20poly1305_afternm(c, m, mlen, n, k); +} + +int +crypto_box_open_afternm(unsigned char *m, const unsigned char *c, + unsigned long long clen, const unsigned char *n, + const unsigned char *k) +{ + return crypto_box_curve25519xsalsa20poly1305_open_afternm(m, c, clen, n, k); +} + +int +crypto_box(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *pk, const unsigned char *sk) +{ + return crypto_box_curve25519xsalsa20poly1305(c, m, mlen, n, pk, sk); +} + +int +crypto_box_open(unsigned char *m, const unsigned char *c, + unsigned long long clen, const unsigned char *n, + const unsigned char *pk, const unsigned char *sk) +{ + return crypto_box_curve25519xsalsa20poly1305_open(m, c, clen, n, pk, sk); +} diff --git a/windows/sodium/crypto_box.h b/sodium/sodium_crypto_box.h similarity index 75% rename from windows/sodium/crypto_box.h rename to sodium/sodium_crypto_box.h index 614cd1e0..8691dab7 100644 --- a/windows/sodium/crypto_box.h +++ b/sodium/sodium_crypto_box.h @@ -10,8 +10,8 @@ #include -#include "crypto_box_curve25519xsalsa20poly1305.h" -#include "export.h" +#include "sodium_crypto_box_curve25519xsalsa20poly1305.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -21,146 +21,126 @@ extern "C" { #endif #define crypto_box_SEEDBYTES crypto_box_curve25519xsalsa20poly1305_SEEDBYTES -SODIUM_EXPORT size_t crypto_box_seedbytes(void); #define crypto_box_PUBLICKEYBYTES crypto_box_curve25519xsalsa20poly1305_PUBLICKEYBYTES -SODIUM_EXPORT size_t crypto_box_publickeybytes(void); #define crypto_box_SECRETKEYBYTES crypto_box_curve25519xsalsa20poly1305_SECRETKEYBYTES -SODIUM_EXPORT size_t crypto_box_secretkeybytes(void); #define crypto_box_NONCEBYTES crypto_box_curve25519xsalsa20poly1305_NONCEBYTES -SODIUM_EXPORT size_t crypto_box_noncebytes(void); #define crypto_box_MACBYTES crypto_box_curve25519xsalsa20poly1305_MACBYTES -SODIUM_EXPORT size_t crypto_box_macbytes(void); +#define crypto_box_MESSAGEBYTES_MAX crypto_box_curve25519xsalsa20poly1305_MESSAGEBYTES_MAX +size_t crypto_box_messagebytes_max(void); + #define crypto_box_PRIMITIVE "curve25519xsalsa20poly1305" -SODIUM_EXPORT const char *crypto_box_primitive(void); -SODIUM_EXPORT int crypto_box_seed_keypair(unsigned char *pk, unsigned char *sk, - const unsigned char *seed); + const unsigned char *seed) + __attribute__ ((nonnull)); -SODIUM_EXPORT -int crypto_box_keypair(unsigned char *pk, unsigned char *sk); +int crypto_box_keypair(unsigned char *pk, unsigned char *sk) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_easy(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_open_easy(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5, 6))); -SODIUM_EXPORT int crypto_box_detached(unsigned char *c, unsigned char *mac, const unsigned char *m, unsigned long long mlen, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_open_detached(unsigned char *m, const unsigned char *c, const unsigned char *mac, unsigned long long clen, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 3, 5, 6, 7))); /* -- Precomputation interface -- */ #define crypto_box_BEFORENMBYTES crypto_box_curve25519xsalsa20poly1305_BEFORENMBYTES -SODIUM_EXPORT size_t crypto_box_beforenmbytes(void); -SODIUM_EXPORT int crypto_box_beforenm(unsigned char *k, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_easy_afternm(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_open_easy_afternm(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5))); -SODIUM_EXPORT int crypto_box_detached_afternm(unsigned char *c, unsigned char *mac, const unsigned char *m, unsigned long long mlen, - const unsigned char *n, const unsigned char *k); + const unsigned char *n, const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_open_detached_afternm(unsigned char *m, const unsigned char *c, const unsigned char *mac, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 3, 5, 6))); /* -- Ephemeral SK interface -- */ #define crypto_box_SEALBYTES (crypto_box_PUBLICKEYBYTES + crypto_box_MACBYTES) -SODIUM_EXPORT size_t crypto_box_sealbytes(void); -SODIUM_EXPORT int crypto_box_seal(unsigned char *c, const unsigned char *m, - unsigned long long mlen, const unsigned char *pk); + unsigned long long mlen, const unsigned char *pk) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_seal_open(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5))); /* -- NaCl compatibility interface ; Requires padding -- */ #define crypto_box_ZEROBYTES crypto_box_curve25519xsalsa20poly1305_ZEROBYTES -SODIUM_EXPORT size_t crypto_box_zerobytes(void); #define crypto_box_BOXZEROBYTES crypto_box_curve25519xsalsa20poly1305_BOXZEROBYTES -SODIUM_EXPORT size_t crypto_box_boxzerobytes(void); -SODIUM_EXPORT int crypto_box(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_open(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5, 6))); -SODIUM_EXPORT int crypto_box_afternm(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_open_afternm(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5))); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_box_curve25519xchacha20poly1305.h b/sodium/sodium_crypto_box_curve25519xchacha20poly1305.h similarity index 83% rename from windows/sodium/crypto_box_curve25519xchacha20poly1305.h rename to sodium/sodium_crypto_box_curve25519xchacha20poly1305.h index b781cc6e..dfed3ce7 100644 --- a/windows/sodium/crypto_box_curve25519xchacha20poly1305.h +++ b/sodium/sodium_crypto_box_curve25519xchacha20poly1305.h @@ -3,7 +3,8 @@ #define crypto_box_curve25519xchacha20poly1305_H #include -#include "export.h" +#include "sodium_crypto_stream_xchacha20.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -13,57 +14,52 @@ extern "C" { #endif #define crypto_box_curve25519xchacha20poly1305_SEEDBYTES 32U -SODIUM_EXPORT size_t crypto_box_curve25519xchacha20poly1305_seedbytes(void); #define crypto_box_curve25519xchacha20poly1305_PUBLICKEYBYTES 32U -SODIUM_EXPORT size_t crypto_box_curve25519xchacha20poly1305_publickeybytes(void); #define crypto_box_curve25519xchacha20poly1305_SECRETKEYBYTES 32U -SODIUM_EXPORT size_t crypto_box_curve25519xchacha20poly1305_secretkeybytes(void); #define crypto_box_curve25519xchacha20poly1305_BEFORENMBYTES 32U -SODIUM_EXPORT size_t crypto_box_curve25519xchacha20poly1305_beforenmbytes(void); #define crypto_box_curve25519xchacha20poly1305_NONCEBYTES 24U -SODIUM_EXPORT size_t crypto_box_curve25519xchacha20poly1305_noncebytes(void); #define crypto_box_curve25519xchacha20poly1305_MACBYTES 16U -SODIUM_EXPORT size_t crypto_box_curve25519xchacha20poly1305_macbytes(void); -SODIUM_EXPORT +#define crypto_box_curve25519xchacha20poly1305_MESSAGEBYTES_MAX \ + (crypto_stream_xchacha20_MESSAGEBYTES_MAX - crypto_box_curve25519xchacha20poly1305_MACBYTES) +size_t crypto_box_curve25519xchacha20poly1305_messagebytes_max(void); + int crypto_box_curve25519xchacha20poly1305_seed_keypair(unsigned char *pk, unsigned char *sk, - const unsigned char *seed); + const unsigned char *seed) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_keypair(unsigned char *pk, - unsigned char *sk); + unsigned char *sk) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_easy(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_open_easy(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5, 6))); -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_detached(unsigned char *c, unsigned char *mac, const unsigned char *m, @@ -71,9 +67,8 @@ int crypto_box_curve25519xchacha20poly1305_detached(unsigned char *c, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_open_detached(unsigned char *m, const unsigned char *c, const unsigned char *mac, @@ -81,47 +76,44 @@ int crypto_box_curve25519xchacha20poly1305_open_detached(unsigned char *m, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 3, 5, 6, 7))); /* -- Precomputation interface -- */ -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_beforenm(unsigned char *k, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_easy_afternm(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_open_easy_afternm(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5))); -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_detached_afternm(unsigned char *c, unsigned char *mac, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_open_detached_afternm(unsigned char *m, const unsigned char *c, const unsigned char *mac, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 3, 5, 6))); /* -- Ephemeral SK interface -- */ @@ -129,22 +121,20 @@ int crypto_box_curve25519xchacha20poly1305_open_detached_afternm(unsigned char * (crypto_box_curve25519xchacha20poly1305_PUBLICKEYBYTES + \ crypto_box_curve25519xchacha20poly1305_MACBYTES) -SODIUM_EXPORT size_t crypto_box_curve25519xchacha20poly1305_sealbytes(void); -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_seal(unsigned char *c, const unsigned char *m, unsigned long long mlen, - const unsigned char *pk); + const unsigned char *pk) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_curve25519xchacha20poly1305_seal_open(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5))); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_box_curve25519xsalsa20poly1305.h b/sodium/sodium_crypto_box_curve25519xsalsa20poly1305.h similarity index 79% rename from windows/sodium/crypto_box_curve25519xsalsa20poly1305.h rename to sodium/sodium_crypto_box_curve25519xsalsa20poly1305.h index 9b5a39c3..3d7c18bf 100644 --- a/windows/sodium/crypto_box_curve25519xsalsa20poly1305.h +++ b/sodium/sodium_crypto_box_curve25519xsalsa20poly1305.h @@ -2,7 +2,8 @@ #define crypto_box_curve25519xsalsa20poly1305_H #include -#include "export.h" +#include "sodium_crypto_stream_xsalsa20.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -12,86 +13,81 @@ extern "C" { #endif #define crypto_box_curve25519xsalsa20poly1305_SEEDBYTES 32U -SODIUM_EXPORT size_t crypto_box_curve25519xsalsa20poly1305_seedbytes(void); #define crypto_box_curve25519xsalsa20poly1305_PUBLICKEYBYTES 32U -SODIUM_EXPORT size_t crypto_box_curve25519xsalsa20poly1305_publickeybytes(void); #define crypto_box_curve25519xsalsa20poly1305_SECRETKEYBYTES 32U -SODIUM_EXPORT size_t crypto_box_curve25519xsalsa20poly1305_secretkeybytes(void); #define crypto_box_curve25519xsalsa20poly1305_BEFORENMBYTES 32U -SODIUM_EXPORT size_t crypto_box_curve25519xsalsa20poly1305_beforenmbytes(void); #define crypto_box_curve25519xsalsa20poly1305_NONCEBYTES 24U -SODIUM_EXPORT size_t crypto_box_curve25519xsalsa20poly1305_noncebytes(void); #define crypto_box_curve25519xsalsa20poly1305_MACBYTES 16U -SODIUM_EXPORT size_t crypto_box_curve25519xsalsa20poly1305_macbytes(void); +/* Only for the libsodium API - The NaCl compatibility API would require BOXZEROBYTES extra bytes */ +#define crypto_box_curve25519xsalsa20poly1305_MESSAGEBYTES_MAX \ + (crypto_stream_xsalsa20_MESSAGEBYTES_MAX - crypto_box_curve25519xsalsa20poly1305_MACBYTES) +size_t crypto_box_curve25519xsalsa20poly1305_messagebytes_max(void); + +int crypto_box_curve25519xsalsa20poly1305_seed_keypair(unsigned char *pk, + unsigned char *sk, + const unsigned char *seed) + __attribute__ ((nonnull)); + +int crypto_box_curve25519xsalsa20poly1305_keypair(unsigned char *pk, + unsigned char *sk) + __attribute__ ((nonnull)); + +int crypto_box_curve25519xsalsa20poly1305_beforenm(unsigned char *k, + const unsigned char *pk, + const unsigned char *sk) + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); + +/* -- NaCl compatibility interface ; Requires padding -- */ + #define crypto_box_curve25519xsalsa20poly1305_BOXZEROBYTES 16U -SODIUM_EXPORT size_t crypto_box_curve25519xsalsa20poly1305_boxzerobytes(void); #define crypto_box_curve25519xsalsa20poly1305_ZEROBYTES \ (crypto_box_curve25519xsalsa20poly1305_BOXZEROBYTES + \ crypto_box_curve25519xsalsa20poly1305_MACBYTES) -SODIUM_EXPORT size_t crypto_box_curve25519xsalsa20poly1305_zerobytes(void); -SODIUM_EXPORT int crypto_box_curve25519xsalsa20poly1305(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_curve25519xsalsa20poly1305_open(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *pk, const unsigned char *sk) - __attribute__ ((warn_unused_result)); - -SODIUM_EXPORT -int crypto_box_curve25519xsalsa20poly1305_seed_keypair(unsigned char *pk, - unsigned char *sk, - const unsigned char *seed); - -SODIUM_EXPORT -int crypto_box_curve25519xsalsa20poly1305_keypair(unsigned char *pk, - unsigned char *sk); - -SODIUM_EXPORT -int crypto_box_curve25519xsalsa20poly1305_beforenm(unsigned char *k, - const unsigned char *pk, - const unsigned char *sk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5, 6))); -SODIUM_EXPORT int crypto_box_curve25519xsalsa20poly1305_afternm(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_box_curve25519xsalsa20poly1305_open_afternm(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5))); #ifdef __cplusplus } diff --git a/sodium/sodium_crypto_box_easy.c b/sodium/sodium_crypto_box_easy.c new file mode 100644 index 00000000..0029bb7b --- /dev/null +++ b/sodium/sodium_crypto_box_easy.c @@ -0,0 +1,115 @@ + +#include +#include +#include + +#include "sodium_core.h" +#include "sodium_crypto_box.h" +#include "sodium_crypto_secretbox.h" +#include "sodium_private_common.h" +#include "sodium_utils.h" + +int +crypto_box_detached_afternm(unsigned char *c, unsigned char *mac, + const unsigned char *m, unsigned long long mlen, + const unsigned char *n, const unsigned char *k) +{ + return crypto_secretbox_detached(c, mac, m, mlen, n, k); +} + +int +crypto_box_detached(unsigned char *c, unsigned char *mac, + const unsigned char *m, unsigned long long mlen, + const unsigned char *n, const unsigned char *pk, + const unsigned char *sk) +{ + unsigned char k[crypto_box_BEFORENMBYTES]; + int ret; + + COMPILER_ASSERT(crypto_box_BEFORENMBYTES >= crypto_secretbox_KEYBYTES); + if (crypto_box_beforenm(k, pk, sk) != 0) { + return -1; + } + ret = crypto_box_detached_afternm(c, mac, m, mlen, n, k); + sodium_memzero(k, sizeof k); + + return ret; +} + +int +crypto_box_easy_afternm(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *k) +{ + if (mlen > crypto_box_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return crypto_box_detached_afternm(c + crypto_box_MACBYTES, c, m, mlen, n, + k); +} + +int +crypto_box_easy(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *pk, const unsigned char *sk) +{ + if (mlen > crypto_box_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return crypto_box_detached(c + crypto_box_MACBYTES, c, m, mlen, n, + pk, sk); +} + +int +crypto_box_open_detached_afternm(unsigned char *m, const unsigned char *c, + const unsigned char *mac, + unsigned long long clen, + const unsigned char *n, + const unsigned char *k) +{ + return crypto_secretbox_open_detached(m, c, mac, clen, n, k); +} + +int +crypto_box_open_detached(unsigned char *m, const unsigned char *c, + const unsigned char *mac, + unsigned long long clen, const unsigned char *n, + const unsigned char *pk, const unsigned char *sk) +{ + unsigned char k[crypto_box_BEFORENMBYTES]; + int ret; + + if (crypto_box_beforenm(k, pk, sk) != 0) { + return -1; + } + ret = crypto_box_open_detached_afternm(m, c, mac, clen, n, k); + sodium_memzero(k, sizeof k); + + return ret; +} + +int +crypto_box_open_easy_afternm(unsigned char *m, const unsigned char *c, + unsigned long long clen, const unsigned char *n, + const unsigned char *k) +{ + if (clen < crypto_box_MACBYTES) { + return -1; + } + return crypto_box_open_detached_afternm(m, c + crypto_box_MACBYTES, c, + clen - crypto_box_MACBYTES, + n, k); +} + +int +crypto_box_open_easy(unsigned char *m, const unsigned char *c, + unsigned long long clen, const unsigned char *n, + const unsigned char *pk, const unsigned char *sk) +{ + if (clen < crypto_box_MACBYTES) { + return -1; + } + return crypto_box_open_detached(m, c + crypto_box_MACBYTES, c, + clen - crypto_box_MACBYTES, + n, pk, sk); +} diff --git a/sodium/sodium_crypto_core_ed25519.h b/sodium/sodium_crypto_core_ed25519.h new file mode 100644 index 00000000..688500f0 --- /dev/null +++ b/sodium/sodium_crypto_core_ed25519.h @@ -0,0 +1,68 @@ +#ifndef crypto_core_ed25519_H +#define crypto_core_ed25519_H + +#include +#include "sodium_export.h" + +#ifdef __cplusplus +extern "C" { +#endif + +#define crypto_core_ed25519_BYTES 32 +size_t crypto_core_ed25519_bytes(void); + +#define crypto_core_ed25519_UNIFORMBYTES 32 +size_t crypto_core_ed25519_uniformbytes(void); + +#define crypto_core_ed25519_SCALARBYTES 32 +size_t crypto_core_ed25519_scalarbytes(void); + +#define crypto_core_ed25519_NONREDUCEDSCALARBYTES 64 +size_t crypto_core_ed25519_nonreducedscalarbytes(void); + +int crypto_core_ed25519_is_valid_point(const unsigned char *p) + __attribute__ ((nonnull)); + +int crypto_core_ed25519_add(unsigned char *r, + const unsigned char *p, const unsigned char *q) + __attribute__ ((nonnull)); + +int crypto_core_ed25519_sub(unsigned char *r, + const unsigned char *p, const unsigned char *q) + __attribute__ ((nonnull)); + +int crypto_core_ed25519_from_uniform(unsigned char *p, const unsigned char *r) + __attribute__ ((nonnull)); + +void crypto_core_ed25519_scalar_random(unsigned char *r) + __attribute__ ((nonnull)); + +int crypto_core_ed25519_scalar_invert(unsigned char *recip, const unsigned char *s) + __attribute__ ((nonnull)); + +void crypto_core_ed25519_scalar_negate(unsigned char *neg, const unsigned char *s) + __attribute__ ((nonnull)); + +void crypto_core_ed25519_scalar_complement(unsigned char *comp, const unsigned char *s) + __attribute__ ((nonnull)); + +void crypto_core_ed25519_scalar_add(unsigned char *z, const unsigned char *x, + const unsigned char *y) + __attribute__ ((nonnull)); + +void crypto_core_ed25519_scalar_sub(unsigned char *z, const unsigned char *x, + const unsigned char *y) + __attribute__ ((nonnull)); + +/* + * The interval `s` is sampled from should be at least 317 bits to ensure almost + * uniformity of `r` over `L`. + */ +void crypto_core_ed25519_scalar_reduce(unsigned char *r, const unsigned char *s) + __attribute__ ((nonnull)); + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/windows/sodium/crypto_core_hchacha20.h b/sodium/sodium_crypto_core_hchacha20.h similarity index 86% rename from windows/sodium/crypto_core_hchacha20.h rename to sodium/sodium_crypto_core_hchacha20.h index 05e5670c..5e462338 100644 --- a/windows/sodium/crypto_core_hchacha20.h +++ b/sodium/sodium_crypto_core_hchacha20.h @@ -2,31 +2,27 @@ #define crypto_core_hchacha20_H #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif #define crypto_core_hchacha20_OUTPUTBYTES 32U -SODIUM_EXPORT size_t crypto_core_hchacha20_outputbytes(void); #define crypto_core_hchacha20_INPUTBYTES 16U -SODIUM_EXPORT size_t crypto_core_hchacha20_inputbytes(void); #define crypto_core_hchacha20_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_core_hchacha20_keybytes(void); #define crypto_core_hchacha20_CONSTBYTES 16U -SODIUM_EXPORT size_t crypto_core_hchacha20_constbytes(void); -SODIUM_EXPORT int crypto_core_hchacha20(unsigned char *out, const unsigned char *in, - const unsigned char *k, const unsigned char *c); + const unsigned char *k, const unsigned char *c) + __attribute__ ((nonnull(1, 2, 3))); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_core_hsalsa20.h b/sodium/sodium_crypto_core_hsalsa20.h similarity index 86% rename from windows/sodium/crypto_core_hsalsa20.h rename to sodium/sodium_crypto_core_hsalsa20.h index 82e475b8..1ee628f8 100644 --- a/windows/sodium/crypto_core_hsalsa20.h +++ b/sodium/sodium_crypto_core_hsalsa20.h @@ -2,31 +2,27 @@ #define crypto_core_hsalsa20_H #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif #define crypto_core_hsalsa20_OUTPUTBYTES 32U -SODIUM_EXPORT size_t crypto_core_hsalsa20_outputbytes(void); #define crypto_core_hsalsa20_INPUTBYTES 16U -SODIUM_EXPORT size_t crypto_core_hsalsa20_inputbytes(void); #define crypto_core_hsalsa20_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_core_hsalsa20_keybytes(void); #define crypto_core_hsalsa20_CONSTBYTES 16U -SODIUM_EXPORT size_t crypto_core_hsalsa20_constbytes(void); -SODIUM_EXPORT int crypto_core_hsalsa20(unsigned char *out, const unsigned char *in, - const unsigned char *k, const unsigned char *c); + const unsigned char *k, const unsigned char *c) + __attribute__ ((nonnull(1, 2, 3))); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_core_salsa20.h b/sodium/sodium_crypto_core_salsa20.h similarity index 86% rename from windows/sodium/crypto_core_salsa20.h rename to sodium/sodium_crypto_core_salsa20.h index 160cc56d..d77b3278 100644 --- a/windows/sodium/crypto_core_salsa20.h +++ b/sodium/sodium_crypto_core_salsa20.h @@ -2,31 +2,27 @@ #define crypto_core_salsa20_H #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif #define crypto_core_salsa20_OUTPUTBYTES 64U -SODIUM_EXPORT size_t crypto_core_salsa20_outputbytes(void); #define crypto_core_salsa20_INPUTBYTES 16U -SODIUM_EXPORT size_t crypto_core_salsa20_inputbytes(void); #define crypto_core_salsa20_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_core_salsa20_keybytes(void); #define crypto_core_salsa20_CONSTBYTES 16U -SODIUM_EXPORT size_t crypto_core_salsa20_constbytes(void); -SODIUM_EXPORT int crypto_core_salsa20(unsigned char *out, const unsigned char *in, - const unsigned char *k, const unsigned char *c); + const unsigned char *k, const unsigned char *c) + __attribute__ ((nonnull(1, 2, 3))); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_core_salsa2012.h b/sodium/sodium_crypto_core_salsa2012.h similarity index 86% rename from windows/sodium/crypto_core_salsa2012.h rename to sodium/sodium_crypto_core_salsa2012.h index bdd5f9fd..3677d5dc 100644 --- a/windows/sodium/crypto_core_salsa2012.h +++ b/sodium/sodium_crypto_core_salsa2012.h @@ -2,31 +2,27 @@ #define crypto_core_salsa2012_H #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif #define crypto_core_salsa2012_OUTPUTBYTES 64U -SODIUM_EXPORT size_t crypto_core_salsa2012_outputbytes(void); #define crypto_core_salsa2012_INPUTBYTES 16U -SODIUM_EXPORT size_t crypto_core_salsa2012_inputbytes(void); #define crypto_core_salsa2012_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_core_salsa2012_keybytes(void); #define crypto_core_salsa2012_CONSTBYTES 16U -SODIUM_EXPORT size_t crypto_core_salsa2012_constbytes(void); -SODIUM_EXPORT int crypto_core_salsa2012(unsigned char *out, const unsigned char *in, - const unsigned char *k, const unsigned char *c); + const unsigned char *k, const unsigned char *c) + __attribute__ ((nonnull(1, 2, 3))); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_core_salsa208.h b/sodium/sodium_crypto_core_salsa208.h similarity index 52% rename from windows/sodium/crypto_core_salsa208.h rename to sodium/sodium_crypto_core_salsa208.h index 3c13efa4..3d9640f5 100644 --- a/windows/sodium/crypto_core_salsa208.h +++ b/sodium/sodium_crypto_core_salsa208.h @@ -2,31 +2,31 @@ #define crypto_core_salsa208_H #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif #define crypto_core_salsa208_OUTPUTBYTES 64U -SODIUM_EXPORT -size_t crypto_core_salsa208_outputbytes(void); +size_t crypto_core_salsa208_outputbytes(void) + __attribute__ ((deprecated)); #define crypto_core_salsa208_INPUTBYTES 16U -SODIUM_EXPORT -size_t crypto_core_salsa208_inputbytes(void); +size_t crypto_core_salsa208_inputbytes(void) + __attribute__ ((deprecated)); #define crypto_core_salsa208_KEYBYTES 32U -SODIUM_EXPORT -size_t crypto_core_salsa208_keybytes(void); +size_t crypto_core_salsa208_keybytes(void) + __attribute__ ((deprecated)); #define crypto_core_salsa208_CONSTBYTES 16U -SODIUM_EXPORT -size_t crypto_core_salsa208_constbytes(void); +size_t crypto_core_salsa208_constbytes(void) + __attribute__ ((deprecated)); -SODIUM_EXPORT int crypto_core_salsa208(unsigned char *out, const unsigned char *in, - const unsigned char *k, const unsigned char *c); + const unsigned char *k, const unsigned char *c) + __attribute__ ((nonnull(1, 2, 3))); #ifdef __cplusplus } diff --git a/sodium/sodium_crypto_generichash.c b/sodium/sodium_crypto_generichash.c new file mode 100644 index 00000000..afa151d1 --- /dev/null +++ b/sodium/sodium_crypto_generichash.c @@ -0,0 +1,91 @@ + +#include "sodium_crypto_generichash.h" +#include "sodium_randombytes.h" + +size_t +crypto_generichash_bytes_min(void) +{ + return crypto_generichash_BYTES_MIN; +} + +size_t +crypto_generichash_bytes_max(void) +{ + return crypto_generichash_BYTES_MAX; +} + +size_t +crypto_generichash_bytes(void) +{ + return crypto_generichash_BYTES; +} + +size_t +crypto_generichash_keybytes_min(void) +{ + return crypto_generichash_KEYBYTES_MIN; +} + +size_t +crypto_generichash_keybytes_max(void) +{ + return crypto_generichash_KEYBYTES_MAX; +} + +size_t +crypto_generichash_keybytes(void) +{ + return crypto_generichash_KEYBYTES; +} + +const char * +crypto_generichash_primitive(void) +{ + return crypto_generichash_PRIMITIVE; +} + +size_t +crypto_generichash_statebytes(void) +{ + return (sizeof(crypto_generichash_state) + (size_t) 63U) & ~(size_t) 63U; +} + +int +crypto_generichash(unsigned char *out, size_t outlen, const unsigned char *in, + unsigned long long inlen, const unsigned char *key, + size_t keylen) +{ + return crypto_generichash_blake2b(out, outlen, in, inlen, key, keylen); +} + +int +crypto_generichash_init(crypto_generichash_state *state, + const unsigned char *key, + const size_t keylen, const size_t outlen) +{ + return crypto_generichash_blake2b_init + ((crypto_generichash_blake2b_state *) state, key, keylen, outlen); +} + +int +crypto_generichash_update(crypto_generichash_state *state, + const unsigned char *in, + unsigned long long inlen) +{ + return crypto_generichash_blake2b_update + ((crypto_generichash_blake2b_state *) state, in, inlen); +} + +int +crypto_generichash_final(crypto_generichash_state *state, + unsigned char *out, const size_t outlen) +{ + return crypto_generichash_blake2b_final + ((crypto_generichash_blake2b_state *) state, out, outlen); +} + +void +crypto_generichash_keygen(unsigned char k[crypto_generichash_KEYBYTES]) +{ + randombytes_buf(k, crypto_generichash_KEYBYTES); +} diff --git a/windows/sodium/crypto_generichash.h b/sodium/sodium_crypto_generichash.h similarity index 76% rename from windows/sodium/crypto_generichash.h rename to sodium/sodium_crypto_generichash.h index 2398fb9d..3882c08a 100644 --- a/windows/sodium/crypto_generichash.h +++ b/sodium/sodium_crypto_generichash.h @@ -3,8 +3,8 @@ #include -#include "crypto_generichash_blake2b.h" -#include "export.h" +#include "sodium_crypto_generichash_blake2b.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -14,59 +14,55 @@ extern "C" { #endif #define crypto_generichash_BYTES_MIN crypto_generichash_blake2b_BYTES_MIN -SODIUM_EXPORT size_t crypto_generichash_bytes_min(void); #define crypto_generichash_BYTES_MAX crypto_generichash_blake2b_BYTES_MAX -SODIUM_EXPORT size_t crypto_generichash_bytes_max(void); #define crypto_generichash_BYTES crypto_generichash_blake2b_BYTES -SODIUM_EXPORT size_t crypto_generichash_bytes(void); #define crypto_generichash_KEYBYTES_MIN crypto_generichash_blake2b_KEYBYTES_MIN -SODIUM_EXPORT size_t crypto_generichash_keybytes_min(void); #define crypto_generichash_KEYBYTES_MAX crypto_generichash_blake2b_KEYBYTES_MAX -SODIUM_EXPORT size_t crypto_generichash_keybytes_max(void); #define crypto_generichash_KEYBYTES crypto_generichash_blake2b_KEYBYTES -SODIUM_EXPORT size_t crypto_generichash_keybytes(void); #define crypto_generichash_PRIMITIVE "blake2b" -SODIUM_EXPORT const char *crypto_generichash_primitive(void); +/* + * Important when writing bindings for other programming languages: + * the state address should be 64-bytes aligned. + */ typedef crypto_generichash_blake2b_state crypto_generichash_state; -SODIUM_EXPORT size_t crypto_generichash_statebytes(void); -SODIUM_EXPORT int crypto_generichash(unsigned char *out, size_t outlen, const unsigned char *in, unsigned long long inlen, - const unsigned char *key, size_t keylen); + const unsigned char *key, size_t keylen) + __attribute__ ((nonnull(1))); -SODIUM_EXPORT int crypto_generichash_init(crypto_generichash_state *state, const unsigned char *key, - const size_t keylen, const size_t outlen); + const size_t keylen, const size_t outlen) + __attribute__ ((nonnull(1))); -SODIUM_EXPORT int crypto_generichash_update(crypto_generichash_state *state, const unsigned char *in, - unsigned long long inlen); + unsigned long long inlen) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_generichash_final(crypto_generichash_state *state, - unsigned char *out, const size_t outlen); + unsigned char *out, const size_t outlen) + __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_generichash_keygen(unsigned char k[crypto_generichash_KEYBYTES]); +void crypto_generichash_keygen(unsigned char k[crypto_generichash_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_generichash_blake2b.h b/sodium/sodium_crypto_generichash_blake2b.h similarity index 83% rename from windows/sodium/crypto_generichash_blake2b.h rename to sodium/sodium_crypto_generichash_blake2b.h index f275625f..942342aa 100644 --- a/windows/sodium/crypto_generichash_blake2b.h +++ b/sodium/sodium_crypto_generichash_blake2b.h @@ -5,7 +5,7 @@ #include #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -20,17 +20,8 @@ extern "C" { # pragma pack(push, 1) #endif -#if defined(_MSC_VER) -#pragma warning( disable : 4324 ) -#endif - typedef struct CRYPTO_ALIGN(64) crypto_generichash_blake2b_state { - uint64_t h[8]; - uint64_t t[2]; - uint64_t f[2]; - uint8_t buf[2 * 128]; - size_t buflen; - uint8_t last_node; + unsigned char opaque[384]; } crypto_generichash_blake2b_state; #if defined(__IBMC__) || defined(__SUNPRO_C) || defined(__SUNPRO_CC) @@ -40,79 +31,69 @@ typedef struct CRYPTO_ALIGN(64) crypto_generichash_blake2b_state { #endif #define crypto_generichash_blake2b_BYTES_MIN 16U -SODIUM_EXPORT size_t crypto_generichash_blake2b_bytes_min(void); #define crypto_generichash_blake2b_BYTES_MAX 64U -SODIUM_EXPORT size_t crypto_generichash_blake2b_bytes_max(void); #define crypto_generichash_blake2b_BYTES 32U -SODIUM_EXPORT size_t crypto_generichash_blake2b_bytes(void); #define crypto_generichash_blake2b_KEYBYTES_MIN 16U -SODIUM_EXPORT size_t crypto_generichash_blake2b_keybytes_min(void); #define crypto_generichash_blake2b_KEYBYTES_MAX 64U -SODIUM_EXPORT size_t crypto_generichash_blake2b_keybytes_max(void); #define crypto_generichash_blake2b_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_generichash_blake2b_keybytes(void); #define crypto_generichash_blake2b_SALTBYTES 16U -SODIUM_EXPORT size_t crypto_generichash_blake2b_saltbytes(void); #define crypto_generichash_blake2b_PERSONALBYTES 16U -SODIUM_EXPORT size_t crypto_generichash_blake2b_personalbytes(void); -SODIUM_EXPORT size_t crypto_generichash_blake2b_statebytes(void); -SODIUM_EXPORT int crypto_generichash_blake2b(unsigned char *out, size_t outlen, const unsigned char *in, unsigned long long inlen, - const unsigned char *key, size_t keylen); + const unsigned char *key, size_t keylen) + __attribute__ ((nonnull(1))); -SODIUM_EXPORT int crypto_generichash_blake2b_salt_personal(unsigned char *out, size_t outlen, const unsigned char *in, unsigned long long inlen, const unsigned char *key, size_t keylen, const unsigned char *salt, - const unsigned char *personal); + const unsigned char *personal) + __attribute__ ((nonnull(1))); -SODIUM_EXPORT int crypto_generichash_blake2b_init(crypto_generichash_blake2b_state *state, const unsigned char *key, - const size_t keylen, const size_t outlen); + const size_t keylen, const size_t outlen) + __attribute__ ((nonnull(1))); -SODIUM_EXPORT int crypto_generichash_blake2b_init_salt_personal(crypto_generichash_blake2b_state *state, const unsigned char *key, const size_t keylen, const size_t outlen, const unsigned char *salt, - const unsigned char *personal); + const unsigned char *personal) + __attribute__ ((nonnull(1))); -SODIUM_EXPORT int crypto_generichash_blake2b_update(crypto_generichash_blake2b_state *state, const unsigned char *in, - unsigned long long inlen); + unsigned long long inlen) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_generichash_blake2b_final(crypto_generichash_blake2b_state *state, unsigned char *out, - const size_t outlen); + const size_t outlen) __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_generichash_blake2b_keygen(unsigned char k[crypto_generichash_blake2b_KEYBYTES]); +void crypto_generichash_blake2b_keygen(unsigned char k[crypto_generichash_blake2b_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_hash.h b/sodium/sodium_crypto_hash.h similarity index 84% rename from windows/sodium/crypto_hash.h rename to sodium/sodium_crypto_hash.h index 302ed5c5..4a9dbc1b 100644 --- a/windows/sodium/crypto_hash.h +++ b/sodium/sodium_crypto_hash.h @@ -10,8 +10,8 @@ #include -#include "crypto_hash_sha512.h" -#include "export.h" +#include "sodium_crypto_hash_sha512.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -21,15 +21,12 @@ extern "C" { #endif #define crypto_hash_BYTES crypto_hash_sha512_BYTES -SODIUM_EXPORT size_t crypto_hash_bytes(void); -SODIUM_EXPORT int crypto_hash(unsigned char *out, const unsigned char *in, - unsigned long long inlen); + unsigned long long inlen) __attribute__ ((nonnull)); #define crypto_hash_PRIMITIVE "sha512" -SODIUM_EXPORT const char *crypto_hash_primitive(void) __attribute__ ((warn_unused_result)); diff --git a/windows/sodium/crypto_hash_sha512.h b/sodium/sodium_crypto_hash_sha512.h similarity index 72% rename from windows/sodium/crypto_hash_sha512.h rename to sodium/sodium_crypto_hash_sha512.h index 6b0330f1..90253187 100644 --- a/windows/sodium/crypto_hash_sha512.h +++ b/sodium/sodium_crypto_hash_sha512.h @@ -12,7 +12,7 @@ #include #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -27,28 +27,25 @@ typedef struct crypto_hash_sha512_state { uint8_t buf[128]; } crypto_hash_sha512_state; -SODIUM_EXPORT size_t crypto_hash_sha512_statebytes(void); #define crypto_hash_sha512_BYTES 64U -SODIUM_EXPORT size_t crypto_hash_sha512_bytes(void); -SODIUM_EXPORT int crypto_hash_sha512(unsigned char *out, const unsigned char *in, - unsigned long long inlen); + unsigned long long inlen) __attribute__ ((nonnull)); -SODIUM_EXPORT -int crypto_hash_sha512_init(crypto_hash_sha512_state *state); +int crypto_hash_sha512_init(crypto_hash_sha512_state *state) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_hash_sha512_update(crypto_hash_sha512_state *state, const unsigned char *in, - unsigned long long inlen); + unsigned long long inlen) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_hash_sha512_final(crypto_hash_sha512_state *state, - unsigned char *out); + unsigned char *out) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/sodium/sodium_crypto_kx.c b/sodium/sodium_crypto_kx.c new file mode 100644 index 00000000..36da2ef2 --- /dev/null +++ b/sodium/sodium_crypto_kx.c @@ -0,0 +1,103 @@ + +#include + +#include "sodium_core.h" +#include "sodium_crypto_generichash.h" +#include "sodium_crypto_kx.h" +#include "sodium_crypto_scalarmult.h" +#include "sodium_private_common.h" +#include "sodium_randombytes.h" +#include "sodium_utils.h" + +int +crypto_kx_keypair(unsigned char pk[crypto_kx_PUBLICKEYBYTES], + unsigned char sk[crypto_kx_SECRETKEYBYTES]) +{ + COMPILER_ASSERT(crypto_kx_SECRETKEYBYTES == crypto_scalarmult_SCALARBYTES); + COMPILER_ASSERT(crypto_kx_PUBLICKEYBYTES == crypto_scalarmult_BYTES); + + randombytes_buf(sk, crypto_kx_SECRETKEYBYTES); + return crypto_scalarmult_base(pk, sk); +} + +int +crypto_kx_client_session_keys(unsigned char rx[crypto_kx_SESSIONKEYBYTES], + unsigned char tx[crypto_kx_SESSIONKEYBYTES], + const unsigned char client_pk[crypto_kx_PUBLICKEYBYTES], + const unsigned char client_sk[crypto_kx_SECRETKEYBYTES], + const unsigned char server_pk[crypto_kx_PUBLICKEYBYTES]) +{ + crypto_generichash_state h; + unsigned char q[crypto_scalarmult_BYTES]; + unsigned char keys[2 * crypto_kx_SESSIONKEYBYTES]; + int i; + + if (rx == NULL) { + rx = tx; + } + if (tx == NULL) { + tx = rx; + } + if (rx == NULL) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } + if (crypto_scalarmult(q, client_sk, server_pk) != 0) { + return -1; + } + COMPILER_ASSERT(sizeof keys <= crypto_generichash_BYTES_MAX); + crypto_generichash_init(&h, NULL, 0U, sizeof keys); + crypto_generichash_update(&h, q, crypto_scalarmult_BYTES); + sodium_memzero(q, sizeof q); + crypto_generichash_update(&h, client_pk, crypto_kx_PUBLICKEYBYTES); + crypto_generichash_update(&h, server_pk, crypto_kx_PUBLICKEYBYTES); + crypto_generichash_final(&h, keys, sizeof keys); + sodium_memzero(&h, sizeof h); + for (i = 0; i < crypto_kx_SESSIONKEYBYTES; i++) { + rx[i] = keys[i]; /* rx cannot be NULL */ + tx[i] = keys[i + crypto_kx_SESSIONKEYBYTES]; /* tx cannot be NULL */ + } + sodium_memzero(keys, sizeof keys); + + return 0; +} + +int +crypto_kx_server_session_keys(unsigned char rx[crypto_kx_SESSIONKEYBYTES], + unsigned char tx[crypto_kx_SESSIONKEYBYTES], + const unsigned char server_pk[crypto_kx_PUBLICKEYBYTES], + const unsigned char server_sk[crypto_kx_SECRETKEYBYTES], + const unsigned char client_pk[crypto_kx_PUBLICKEYBYTES]) +{ + crypto_generichash_state h; + unsigned char q[crypto_scalarmult_BYTES]; + unsigned char keys[2 * crypto_kx_SESSIONKEYBYTES]; + int i; + + if (rx == NULL) { + rx = tx; + } + if (tx == NULL) { + tx = rx; + } + if (rx == NULL) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } + if (crypto_scalarmult(q, server_sk, client_pk) != 0) { + return -1; + } + COMPILER_ASSERT(sizeof keys <= crypto_generichash_BYTES_MAX); + crypto_generichash_init(&h, NULL, 0U, sizeof keys); + crypto_generichash_update(&h, q, crypto_scalarmult_BYTES); + sodium_memzero(q, sizeof q); + crypto_generichash_update(&h, client_pk, crypto_kx_PUBLICKEYBYTES); + crypto_generichash_update(&h, server_pk, crypto_kx_PUBLICKEYBYTES); + crypto_generichash_final(&h, keys, sizeof keys); + sodium_memzero(&h, sizeof h); + for (i = 0; i < crypto_kx_SESSIONKEYBYTES; i++) { + tx[i] = keys[i]; + rx[i] = keys[i + crypto_kx_SESSIONKEYBYTES]; + } + sodium_memzero(keys, sizeof keys); + + return 0; +} diff --git a/windows/sodium/crypto_kx.h b/sodium/sodium_crypto_kx.h similarity index 86% rename from windows/sodium/crypto_kx.h rename to sodium/sodium_crypto_kx.h index d1fce90d..0f7a9925 100644 --- a/windows/sodium/crypto_kx.h +++ b/sodium/sodium_crypto_kx.h @@ -3,7 +3,7 @@ #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -13,49 +13,42 @@ extern "C" { #endif #define crypto_kx_PUBLICKEYBYTES 32 -SODIUM_EXPORT size_t crypto_kx_publickeybytes(void); #define crypto_kx_SECRETKEYBYTES 32 -SODIUM_EXPORT size_t crypto_kx_secretkeybytes(void); #define crypto_kx_SEEDBYTES 32 -SODIUM_EXPORT size_t crypto_kx_seedbytes(void); #define crypto_kx_SESSIONKEYBYTES 32 -SODIUM_EXPORT size_t crypto_kx_sessionkeybytes(void); #define crypto_kx_PRIMITIVE "x25519blake2b" -SODIUM_EXPORT const char *crypto_kx_primitive(void); -SODIUM_EXPORT int crypto_kx_seed_keypair(unsigned char pk[crypto_kx_PUBLICKEYBYTES], unsigned char sk[crypto_kx_SECRETKEYBYTES], - const unsigned char seed[crypto_kx_SEEDBYTES]); + const unsigned char seed[crypto_kx_SEEDBYTES]) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_kx_keypair(unsigned char pk[crypto_kx_PUBLICKEYBYTES], - unsigned char sk[crypto_kx_SECRETKEYBYTES]); + unsigned char sk[crypto_kx_SECRETKEYBYTES]) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_kx_client_session_keys(unsigned char rx[crypto_kx_SESSIONKEYBYTES], unsigned char tx[crypto_kx_SESSIONKEYBYTES], const unsigned char client_pk[crypto_kx_PUBLICKEYBYTES], const unsigned char client_sk[crypto_kx_SECRETKEYBYTES], const unsigned char server_pk[crypto_kx_PUBLICKEYBYTES]) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(3, 4, 5))); -SODIUM_EXPORT int crypto_kx_server_session_keys(unsigned char rx[crypto_kx_SESSIONKEYBYTES], unsigned char tx[crypto_kx_SESSIONKEYBYTES], const unsigned char server_pk[crypto_kx_PUBLICKEYBYTES], const unsigned char server_sk[crypto_kx_SECRETKEYBYTES], const unsigned char client_pk[crypto_kx_PUBLICKEYBYTES]) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(3, 4, 5))); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_onetimeauth.h b/sodium/sodium_crypto_onetimeauth.h similarity index 71% rename from windows/sodium/crypto_onetimeauth.h rename to sodium/sodium_crypto_onetimeauth.h index 5951c5b8..21fd1ee8 100644 --- a/windows/sodium/crypto_onetimeauth.h +++ b/sodium/sodium_crypto_onetimeauth.h @@ -3,8 +3,8 @@ #include -#include "crypto_onetimeauth_poly1305.h" -#include "export.h" +#include "sodium_crypto_onetimeauth_poly1305.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -15,45 +15,38 @@ extern "C" { typedef crypto_onetimeauth_poly1305_state crypto_onetimeauth_state; -SODIUM_EXPORT size_t crypto_onetimeauth_statebytes(void); #define crypto_onetimeauth_BYTES crypto_onetimeauth_poly1305_BYTES -SODIUM_EXPORT size_t crypto_onetimeauth_bytes(void); #define crypto_onetimeauth_KEYBYTES crypto_onetimeauth_poly1305_KEYBYTES -SODIUM_EXPORT size_t crypto_onetimeauth_keybytes(void); #define crypto_onetimeauth_PRIMITIVE "poly1305" -SODIUM_EXPORT const char *crypto_onetimeauth_primitive(void); -SODIUM_EXPORT int crypto_onetimeauth(unsigned char *out, const unsigned char *in, - unsigned long long inlen, const unsigned char *k); + unsigned long long inlen, const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_onetimeauth_verify(const unsigned char *h, const unsigned char *in, unsigned long long inlen, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_onetimeauth_init(crypto_onetimeauth_state *state, - const unsigned char *key); + const unsigned char *key) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_onetimeauth_update(crypto_onetimeauth_state *state, const unsigned char *in, - unsigned long long inlen); + unsigned long long inlen) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_onetimeauth_final(crypto_onetimeauth_state *state, - unsigned char *out); + unsigned char *out) __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_onetimeauth_keygen(unsigned char k[crypto_onetimeauth_KEYBYTES]); +void crypto_onetimeauth_keygen(unsigned char k[crypto_onetimeauth_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_onetimeauth_poly1305.h b/sodium/sodium_crypto_onetimeauth_poly1305.h similarity index 75% rename from windows/sodium/crypto_onetimeauth_poly1305.h rename to sodium/sodium_crypto_onetimeauth_poly1305.h index 4b89c4f0..1a9e7b3b 100644 --- a/windows/sodium/crypto_onetimeauth_poly1305.h +++ b/sodium/sodium_crypto_onetimeauth_poly1305.h @@ -14,51 +14,47 @@ extern "C" { #include -#include "export.h" +#include "sodium_export.h" typedef struct CRYPTO_ALIGN(16) crypto_onetimeauth_poly1305_state { unsigned char opaque[256]; } crypto_onetimeauth_poly1305_state; -SODIUM_EXPORT size_t crypto_onetimeauth_poly1305_statebytes(void); #define crypto_onetimeauth_poly1305_BYTES 16U -SODIUM_EXPORT size_t crypto_onetimeauth_poly1305_bytes(void); #define crypto_onetimeauth_poly1305_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_onetimeauth_poly1305_keybytes(void); -SODIUM_EXPORT int crypto_onetimeauth_poly1305(unsigned char *out, const unsigned char *in, unsigned long long inlen, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_onetimeauth_poly1305_verify(const unsigned char *h, const unsigned char *in, unsigned long long inlen, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_onetimeauth_poly1305_init(crypto_onetimeauth_poly1305_state *state, - const unsigned char *key); + const unsigned char *key) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_onetimeauth_poly1305_update(crypto_onetimeauth_poly1305_state *state, const unsigned char *in, - unsigned long long inlen); + unsigned long long inlen) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_onetimeauth_poly1305_final(crypto_onetimeauth_poly1305_state *state, - unsigned char *out); + unsigned char *out) + __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_onetimeauth_poly1305_keygen(unsigned char k[crypto_onetimeauth_poly1305_KEYBYTES]); +void crypto_onetimeauth_poly1305_keygen(unsigned char k[crypto_onetimeauth_poly1305_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/sodium/sodium_crypto_scalarmult.c b/sodium/sodium_crypto_scalarmult.c new file mode 100644 index 00000000..d0ac6132 --- /dev/null +++ b/sodium/sodium_crypto_scalarmult.c @@ -0,0 +1,33 @@ + +#include "sodium_crypto_scalarmult.h" + +const char * +crypto_scalarmult_primitive(void) +{ + return crypto_scalarmult_PRIMITIVE; +} + +int +crypto_scalarmult_base(unsigned char *q, const unsigned char *n) +{ + return crypto_scalarmult_curve25519_base(q, n); +} + +int +crypto_scalarmult(unsigned char *q, const unsigned char *n, + const unsigned char *p) +{ + return crypto_scalarmult_curve25519(q, n, p); +} + +size_t +crypto_scalarmult_bytes(void) +{ + return crypto_scalarmult_BYTES; +} + +size_t +crypto_scalarmult_scalarbytes(void) +{ + return crypto_scalarmult_SCALARBYTES; +} diff --git a/windows/sodium/crypto_scalarmult.h b/sodium/sodium_crypto_scalarmult.h similarity index 59% rename from windows/sodium/crypto_scalarmult.h rename to sodium/sodium_crypto_scalarmult.h index 830c10f6..bab5045e 100644 --- a/windows/sodium/crypto_scalarmult.h +++ b/sodium/sodium_crypto_scalarmult.h @@ -3,32 +3,36 @@ #include -#include "crypto_scalarmult_curve25519.h" -#include "export.h" +#include "sodium_crypto_scalarmult_curve25519.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif #define crypto_scalarmult_BYTES crypto_scalarmult_curve25519_BYTES -SODIUM_EXPORT size_t crypto_scalarmult_bytes(void); #define crypto_scalarmult_SCALARBYTES crypto_scalarmult_curve25519_SCALARBYTES -SODIUM_EXPORT size_t crypto_scalarmult_scalarbytes(void); #define crypto_scalarmult_PRIMITIVE "curve25519" -SODIUM_EXPORT const char *crypto_scalarmult_primitive(void); -SODIUM_EXPORT -int crypto_scalarmult_base(unsigned char *q, const unsigned char *n); - -SODIUM_EXPORT +int crypto_scalarmult_base(unsigned char *q, const unsigned char *n) + __attribute__ ((nonnull)); + +/* + * NOTE: Do not use the result of this function directly. + * + * Hash the result with the public keys in order to compute a shared + * secret key: H(q || client_pk || server_pk) + * + * Or unless this is not an option, use the crypto_kx() API instead. + */ int crypto_scalarmult(unsigned char *q, const unsigned char *n, const unsigned char *p) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_scalarmult_curve25519.h b/sodium/sodium_crypto_scalarmult_curve25519.h similarity index 50% rename from windows/sodium/crypto_scalarmult_curve25519.h rename to sodium/sodium_crypto_scalarmult_curve25519.h index d96840c7..24144b4c 100644 --- a/windows/sodium/crypto_scalarmult_curve25519.h +++ b/sodium/sodium_crypto_scalarmult_curve25519.h @@ -3,27 +3,33 @@ #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif #define crypto_scalarmult_curve25519_BYTES 32U -SODIUM_EXPORT size_t crypto_scalarmult_curve25519_bytes(void); #define crypto_scalarmult_curve25519_SCALARBYTES 32U -SODIUM_EXPORT size_t crypto_scalarmult_curve25519_scalarbytes(void); -SODIUM_EXPORT +/* + * NOTE: Do not use the result of this function directly. + * + * Hash the result with the public keys in order to compute a shared + * secret key: H(q || client_pk || server_pk) + * + * Or unless this is not an option, use the crypto_kx() API instead. + */ int crypto_scalarmult_curve25519(unsigned char *q, const unsigned char *n, const unsigned char *p) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT -int crypto_scalarmult_curve25519_base(unsigned char *q, const unsigned char *n); +int crypto_scalarmult_curve25519_base(unsigned char *q, + const unsigned char *n) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/sodium/sodium_crypto_scalarmult_ed25519.h b/sodium/sodium_crypto_scalarmult_ed25519.h new file mode 100644 index 00000000..4ae8f70d --- /dev/null +++ b/sodium/sodium_crypto_scalarmult_ed25519.h @@ -0,0 +1,45 @@ + +#ifndef crypto_scalarmult_ed25519_H +#define crypto_scalarmult_ed25519_H + +#include + +#include "sodium_export.h" + +#ifdef __cplusplus +extern "C" { +#endif + +#define crypto_scalarmult_ed25519_BYTES 32U +size_t crypto_scalarmult_ed25519_bytes(void); + +#define crypto_scalarmult_ed25519_SCALARBYTES 32U +size_t crypto_scalarmult_ed25519_scalarbytes(void); + +/* + * NOTE: Do not use the result of this function directly. + * + * Hash the result with the public keys in order to compute a shared + * secret key: H(q || client_pk || server_pk) + * + * Or unless this is not an option, use the crypto_kx() API instead. + */ +int crypto_scalarmult_ed25519(unsigned char *q, const unsigned char *n, + const unsigned char *p) + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); + +int crypto_scalarmult_ed25519_noclamp(unsigned char *q, const unsigned char *n, + const unsigned char *p) + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); + +int crypto_scalarmult_ed25519_base(unsigned char *q, const unsigned char *n) + __attribute__ ((nonnull)); + +int crypto_scalarmult_ed25519_base_noclamp(unsigned char *q, const unsigned char *n) + __attribute__ ((nonnull)); + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/sodium/sodium_crypto_secretbox.c b/sodium/sodium_crypto_secretbox.c new file mode 100644 index 00000000..5cac0b7f --- /dev/null +++ b/sodium/sodium_crypto_secretbox.c @@ -0,0 +1,25 @@ + +#include "sodium_crypto_secretbox.h" +#include "sodium_randombytes.h" + +int +crypto_secretbox(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *k) +{ + return crypto_secretbox_xsalsa20poly1305(c, m, mlen, n, k); +} + +int +crypto_secretbox_open(unsigned char *m, const unsigned char *c, + unsigned long long clen, const unsigned char *n, + const unsigned char *k) +{ + return crypto_secretbox_xsalsa20poly1305_open(m, c, clen, n, k); +} + +void +crypto_secretbox_keygen(unsigned char k[crypto_secretbox_KEYBYTES]) +{ + randombytes_buf(k, crypto_secretbox_KEYBYTES); +} diff --git a/windows/sodium/crypto_secretbox.h b/sodium/sodium_crypto_secretbox.h similarity index 75% rename from windows/sodium/crypto_secretbox.h rename to sodium/sodium_crypto_secretbox.h index 9b098200..fa9f793e 100644 --- a/windows/sodium/crypto_secretbox.h +++ b/sodium/sodium_crypto_secretbox.h @@ -3,8 +3,8 @@ #include -#include "crypto_secretbox_xsalsa20poly1305.h" -#include "export.h" +#include "sodium_crypto_secretbox_xsalsa20poly1305.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -14,71 +14,63 @@ extern "C" { #endif #define crypto_secretbox_KEYBYTES crypto_secretbox_xsalsa20poly1305_KEYBYTES -SODIUM_EXPORT size_t crypto_secretbox_keybytes(void); #define crypto_secretbox_NONCEBYTES crypto_secretbox_xsalsa20poly1305_NONCEBYTES -SODIUM_EXPORT size_t crypto_secretbox_noncebytes(void); #define crypto_secretbox_MACBYTES crypto_secretbox_xsalsa20poly1305_MACBYTES -SODIUM_EXPORT size_t crypto_secretbox_macbytes(void); #define crypto_secretbox_PRIMITIVE "xsalsa20poly1305" -SODIUM_EXPORT const char *crypto_secretbox_primitive(void); -SODIUM_EXPORT +#define crypto_secretbox_MESSAGEBYTES_MAX crypto_secretbox_xsalsa20poly1305_MESSAGEBYTES_MAX +size_t crypto_secretbox_messagebytes_max(void); + int crypto_secretbox_easy(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_secretbox_open_easy(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5))); -SODIUM_EXPORT int crypto_secretbox_detached(unsigned char *c, unsigned char *mac, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_secretbox_open_detached(unsigned char *m, const unsigned char *c, const unsigned char *mac, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 3, 5, 6))); -SODIUM_EXPORT -void crypto_secretbox_keygen(unsigned char k[crypto_secretbox_KEYBYTES]); +void crypto_secretbox_keygen(unsigned char k[crypto_secretbox_KEYBYTES]) + __attribute__ ((nonnull)); /* -- NaCl compatibility interface ; Requires padding -- */ #define crypto_secretbox_ZEROBYTES crypto_secretbox_xsalsa20poly1305_ZEROBYTES -SODIUM_EXPORT size_t crypto_secretbox_zerobytes(void); #define crypto_secretbox_BOXZEROBYTES crypto_secretbox_xsalsa20poly1305_BOXZEROBYTES -SODIUM_EXPORT size_t crypto_secretbox_boxzerobytes(void); -SODIUM_EXPORT int crypto_secretbox(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_secretbox_open(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5))); #ifdef __cplusplus } diff --git a/sodium/sodium_crypto_secretbox_easy.c b/sodium/sodium_crypto_secretbox_easy.c new file mode 100644 index 00000000..f972225e --- /dev/null +++ b/sodium/sodium_crypto_secretbox_easy.c @@ -0,0 +1,144 @@ + +#include +#include +#include +#include +#include + +#include "sodium_core.h" +#include "sodium_crypto_core_hsalsa20.h" +#include "sodium_crypto_onetimeauth_poly1305.h" +#include "sodium_crypto_secretbox.h" +#include "sodium_crypto_stream_salsa20.h" +#include "sodium_private_common.h" +#include "sodium_utils.h" + +int +crypto_secretbox_detached(unsigned char *c, unsigned char *mac, + const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *k) +{ + crypto_onetimeauth_poly1305_state state; + unsigned char block0[64U]; + unsigned char subkey[crypto_stream_salsa20_KEYBYTES]; + unsigned long long i; + unsigned long long mlen0; + + crypto_core_hsalsa20(subkey, n, k, NULL); + + if (((uintptr_t) c > (uintptr_t) m && + (uintptr_t) c - (uintptr_t) m < mlen) || + ((uintptr_t) m > (uintptr_t) c && + (uintptr_t) m - (uintptr_t) c < mlen)) { /* LCOV_EXCL_LINE */ + memmove(c, m, mlen); + m = c; + } + memset(block0, 0U, crypto_secretbox_ZEROBYTES); + COMPILER_ASSERT(64U >= crypto_secretbox_ZEROBYTES); + mlen0 = mlen; + if (mlen0 > 64U - crypto_secretbox_ZEROBYTES) { + mlen0 = 64U - crypto_secretbox_ZEROBYTES; + } + for (i = 0U; i < mlen0; i++) { + block0[i + crypto_secretbox_ZEROBYTES] = m[i]; + } + crypto_stream_salsa20_xor(block0, block0, + mlen0 + crypto_secretbox_ZEROBYTES, + n + 16, subkey); + COMPILER_ASSERT(crypto_secretbox_ZEROBYTES >= + crypto_onetimeauth_poly1305_KEYBYTES); + crypto_onetimeauth_poly1305_init(&state, block0); + + for (i = 0U; i < mlen0; i++) { + c[i] = block0[crypto_secretbox_ZEROBYTES + i]; + } + sodium_memzero(block0, sizeof block0); + if (mlen > mlen0) { + crypto_stream_salsa20_xor_ic(c + mlen0, m + mlen0, mlen - mlen0, + n + 16, 1U, subkey); + } + sodium_memzero(subkey, sizeof subkey); + + crypto_onetimeauth_poly1305_update(&state, c, mlen); + crypto_onetimeauth_poly1305_final(&state, mac); + sodium_memzero(&state, sizeof state); + + return 0; +} + +int +crypto_secretbox_easy(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *k) +{ + if (mlen > crypto_secretbox_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return crypto_secretbox_detached(c + crypto_secretbox_MACBYTES, + c, m, mlen, n, k); +} + +int +crypto_secretbox_open_detached(unsigned char *m, const unsigned char *c, + const unsigned char *mac, + unsigned long long clen, + const unsigned char *n, + const unsigned char *k) +{ + unsigned char block0[64U]; + unsigned char subkey[crypto_stream_salsa20_KEYBYTES]; + unsigned long long i; + unsigned long long mlen0; + + crypto_core_hsalsa20(subkey, n, k, NULL); + crypto_stream_salsa20(block0, crypto_stream_salsa20_KEYBYTES, + n + 16, subkey); + if (crypto_onetimeauth_poly1305_verify(mac, c, clen, block0) != 0) { + sodium_memzero(subkey, sizeof subkey); + return -1; + } + if (m == NULL) { + return 0; + } + if (((uintptr_t) c >= (uintptr_t) m && + (uintptr_t) c - (uintptr_t) m < clen) || + ((uintptr_t) m >= (uintptr_t) c && + (uintptr_t) m - (uintptr_t) c < clen)) { /* LCOV_EXCL_LINE */ + memmove(m, c, clen); + c = m; + } + mlen0 = clen; + if (mlen0 > 64U - crypto_secretbox_ZEROBYTES) { + mlen0 = 64U - crypto_secretbox_ZEROBYTES; + } + for (i = 0U; i < mlen0; i++) { + block0[crypto_secretbox_ZEROBYTES + i] = c[i]; + } + crypto_stream_salsa20_xor(block0, block0, + crypto_secretbox_ZEROBYTES + mlen0, + n + 16, subkey); + for (i = 0U; i < mlen0; i++) { + m[i] = block0[i + crypto_secretbox_ZEROBYTES]; + } + if (clen > mlen0) { + crypto_stream_salsa20_xor_ic(m + mlen0, c + mlen0, clen - mlen0, + n + 16, 1U, subkey); + } + sodium_memzero(subkey, sizeof subkey); + + return 0; +} + +int +crypto_secretbox_open_easy(unsigned char *m, const unsigned char *c, + unsigned long long clen, const unsigned char *n, + const unsigned char *k) +{ + if (clen < crypto_secretbox_MACBYTES) { + return -1; + } + return crypto_secretbox_open_detached(m, c + crypto_secretbox_MACBYTES, c, + clen - crypto_secretbox_MACBYTES, + n, k); +} diff --git a/windows/sodium/crypto_secretbox_xchacha20poly1305.h b/sodium/sodium_crypto_secretbox_xchacha20poly1305.h similarity index 79% rename from windows/sodium/crypto_secretbox_xchacha20poly1305.h rename to sodium/sodium_crypto_secretbox_xchacha20poly1305.h index 7a61a091..84106690 100644 --- a/windows/sodium/crypto_secretbox_xchacha20poly1305.h +++ b/sodium/sodium_crypto_secretbox_xchacha20poly1305.h @@ -2,7 +2,8 @@ #define crypto_secretbox_xchacha20poly1305_H #include -#include "export.h" +#include "sodium_crypto_stream_xchacha20.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -12,48 +13,47 @@ extern "C" { #endif #define crypto_secretbox_xchacha20poly1305_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_secretbox_xchacha20poly1305_keybytes(void); #define crypto_secretbox_xchacha20poly1305_NONCEBYTES 24U -SODIUM_EXPORT size_t crypto_secretbox_xchacha20poly1305_noncebytes(void); #define crypto_secretbox_xchacha20poly1305_MACBYTES 16U -SODIUM_EXPORT size_t crypto_secretbox_xchacha20poly1305_macbytes(void); -SODIUM_EXPORT +#define crypto_secretbox_xchacha20poly1305_MESSAGEBYTES_MAX \ + (crypto_stream_xchacha20_MESSAGEBYTES_MAX - crypto_secretbox_xchacha20poly1305_MACBYTES) +size_t crypto_secretbox_xchacha20poly1305_messagebytes_max(void); + int crypto_secretbox_xchacha20poly1305_easy(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_secretbox_xchacha20poly1305_open_easy(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5))); -SODIUM_EXPORT int crypto_secretbox_xchacha20poly1305_detached(unsigned char *c, unsigned char *mac, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_secretbox_xchacha20poly1305_open_detached(unsigned char *m, const unsigned char *c, const unsigned char *mac, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 3, 5, 6))); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_secretbox_xsalsa20poly1305.h b/sodium/sodium_crypto_secretbox_xsalsa20poly1305.h similarity index 68% rename from windows/sodium/crypto_secretbox_xsalsa20poly1305.h rename to sodium/sodium_crypto_secretbox_xsalsa20poly1305.h index 5aa30805..01086c93 100644 --- a/windows/sodium/crypto_secretbox_xsalsa20poly1305.h +++ b/sodium/sodium_crypto_secretbox_xsalsa20poly1305.h @@ -2,7 +2,8 @@ #define crypto_secretbox_xsalsa20poly1305_H #include -#include "export.h" +#include "sodium_crypto_stream_xsalsa20.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -12,44 +13,45 @@ extern "C" { #endif #define crypto_secretbox_xsalsa20poly1305_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_secretbox_xsalsa20poly1305_keybytes(void); #define crypto_secretbox_xsalsa20poly1305_NONCEBYTES 24U -SODIUM_EXPORT size_t crypto_secretbox_xsalsa20poly1305_noncebytes(void); #define crypto_secretbox_xsalsa20poly1305_MACBYTES 16U -SODIUM_EXPORT size_t crypto_secretbox_xsalsa20poly1305_macbytes(void); -#define crypto_secretbox_xsalsa20poly1305_BOXZEROBYTES 16U -SODIUM_EXPORT -size_t crypto_secretbox_xsalsa20poly1305_boxzerobytes(void); - -#define crypto_secretbox_xsalsa20poly1305_ZEROBYTES \ - (crypto_secretbox_xsalsa20poly1305_BOXZEROBYTES + \ - crypto_secretbox_xsalsa20poly1305_MACBYTES) -SODIUM_EXPORT -size_t crypto_secretbox_xsalsa20poly1305_zerobytes(void); +/* Only for the libsodium API - The NaCl compatibility API would require BOXZEROBYTES extra bytes */ +#define crypto_secretbox_xsalsa20poly1305_MESSAGEBYTES_MAX \ + (crypto_stream_xsalsa20_MESSAGEBYTES_MAX - crypto_secretbox_xsalsa20poly1305_MACBYTES) +size_t crypto_secretbox_xsalsa20poly1305_messagebytes_max(void); -SODIUM_EXPORT int crypto_secretbox_xsalsa20poly1305(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_secretbox_xsalsa20poly1305_open(unsigned char *m, const unsigned char *c, unsigned long long clen, const unsigned char *n, const unsigned char *k) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(2, 4, 5))); + +void crypto_secretbox_xsalsa20poly1305_keygen(unsigned char k[crypto_secretbox_xsalsa20poly1305_KEYBYTES]) + __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_secretbox_xsalsa20poly1305_keygen(unsigned char k[crypto_secretbox_xsalsa20poly1305_KEYBYTES]); +/* -- NaCl compatibility interface ; Requires padding -- */ + +#define crypto_secretbox_xsalsa20poly1305_BOXZEROBYTES 16U +size_t crypto_secretbox_xsalsa20poly1305_boxzerobytes(void); + +#define crypto_secretbox_xsalsa20poly1305_ZEROBYTES \ + (crypto_secretbox_xsalsa20poly1305_BOXZEROBYTES + \ + crypto_secretbox_xsalsa20poly1305_MACBYTES) +size_t crypto_secretbox_xsalsa20poly1305_zerobytes(void); #ifdef __cplusplus } diff --git a/sodium/sodium_crypto_secretstream_xchacha20poly1305.h b/sodium/sodium_crypto_secretstream_xchacha20poly1305.h new file mode 100644 index 00000000..c8fb19f2 --- /dev/null +++ b/sodium/sodium_crypto_secretstream_xchacha20poly1305.h @@ -0,0 +1,93 @@ +#ifndef crypto_secretstream_xchacha20poly1305_H +#define crypto_secretstream_xchacha20poly1305_H + +#include + +#include "sodium_crypto_aead_xchacha20poly1305.h" +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_export.h" + +#ifdef __cplusplus +# ifdef __GNUC__ +# pragma GCC diagnostic ignored "-Wlong-long" +# endif +extern "C" { +#endif + +#define crypto_secretstream_xchacha20poly1305_ABYTES \ + (1U + crypto_aead_xchacha20poly1305_ietf_ABYTES) +size_t crypto_secretstream_xchacha20poly1305_abytes(void); + +#define crypto_secretstream_xchacha20poly1305_HEADERBYTES \ + crypto_aead_xchacha20poly1305_ietf_NPUBBYTES +size_t crypto_secretstream_xchacha20poly1305_headerbytes(void); + +#define crypto_secretstream_xchacha20poly1305_KEYBYTES \ + crypto_aead_xchacha20poly1305_ietf_KEYBYTES +size_t crypto_secretstream_xchacha20poly1305_keybytes(void); + +#define crypto_secretstream_xchacha20poly1305_MESSAGEBYTES_MAX \ + SODIUM_MIN(SODIUM_SIZE_MAX - crypto_secretstream_xchacha20poly1305_ABYTES, \ + (64ULL * ((1ULL << 32) - 2ULL))) +size_t crypto_secretstream_xchacha20poly1305_messagebytes_max(void); + +#define crypto_secretstream_xchacha20poly1305_TAG_MESSAGE 0x00 +unsigned char crypto_secretstream_xchacha20poly1305_tag_message(void); + +#define crypto_secretstream_xchacha20poly1305_TAG_PUSH 0x01 +unsigned char crypto_secretstream_xchacha20poly1305_tag_push(void); + +#define crypto_secretstream_xchacha20poly1305_TAG_REKEY 0x02 +unsigned char crypto_secretstream_xchacha20poly1305_tag_rekey(void); + +#define crypto_secretstream_xchacha20poly1305_TAG_FINAL \ + (crypto_secretstream_xchacha20poly1305_TAG_PUSH | \ + crypto_secretstream_xchacha20poly1305_TAG_REKEY) +unsigned char crypto_secretstream_xchacha20poly1305_tag_final(void); + +typedef struct crypto_secretstream_xchacha20poly1305_state { + unsigned char k[crypto_stream_chacha20_ietf_KEYBYTES]; + unsigned char nonce[crypto_stream_chacha20_ietf_NONCEBYTES]; + unsigned char _pad[8]; +} crypto_secretstream_xchacha20poly1305_state; + +size_t crypto_secretstream_xchacha20poly1305_statebytes(void); + +void crypto_secretstream_xchacha20poly1305_keygen + (unsigned char k[crypto_secretstream_xchacha20poly1305_KEYBYTES]) + __attribute__ ((nonnull)); + +int crypto_secretstream_xchacha20poly1305_init_push + (crypto_secretstream_xchacha20poly1305_state *state, + unsigned char header[crypto_secretstream_xchacha20poly1305_HEADERBYTES], + const unsigned char k[crypto_secretstream_xchacha20poly1305_KEYBYTES]) + __attribute__ ((nonnull)); + +int crypto_secretstream_xchacha20poly1305_push + (crypto_secretstream_xchacha20poly1305_state *state, + unsigned char *c, unsigned long long *clen_p, + const unsigned char *m, unsigned long long mlen, + const unsigned char *ad, unsigned long long adlen, unsigned char tag) + __attribute__ ((nonnull(1))); + +int crypto_secretstream_xchacha20poly1305_init_pull + (crypto_secretstream_xchacha20poly1305_state *state, + const unsigned char header[crypto_secretstream_xchacha20poly1305_HEADERBYTES], + const unsigned char k[crypto_secretstream_xchacha20poly1305_KEYBYTES]) + __attribute__ ((nonnull)); + +int crypto_secretstream_xchacha20poly1305_pull + (crypto_secretstream_xchacha20poly1305_state *state, + unsigned char *m, unsigned long long *mlen_p, unsigned char *tag_p, + const unsigned char *c, unsigned long long clen, + const unsigned char *ad, unsigned long long adlen) + __attribute__ ((nonnull(1))); + +void crypto_secretstream_xchacha20poly1305_rekey + (crypto_secretstream_xchacha20poly1305_state *state); + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/windows/sodium/crypto_shorthash.h b/sodium/sodium_crypto_shorthash.h similarity index 81% rename from windows/sodium/crypto_shorthash.h rename to sodium/sodium_crypto_shorthash.h index a4988082..539c9c35 100644 --- a/windows/sodium/crypto_shorthash.h +++ b/sodium/sodium_crypto_shorthash.h @@ -3,8 +3,8 @@ #include -#include "crypto_shorthash_siphash24.h" -#include "export.h" +#include "sodium_crypto_shorthash_siphash24.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -14,23 +14,20 @@ extern "C" { #endif #define crypto_shorthash_BYTES crypto_shorthash_siphash24_BYTES -SODIUM_EXPORT size_t crypto_shorthash_bytes(void); #define crypto_shorthash_KEYBYTES crypto_shorthash_siphash24_KEYBYTES -SODIUM_EXPORT size_t crypto_shorthash_keybytes(void); #define crypto_shorthash_PRIMITIVE "siphash24" -SODIUM_EXPORT const char *crypto_shorthash_primitive(void); -SODIUM_EXPORT int crypto_shorthash(unsigned char *out, const unsigned char *in, - unsigned long long inlen, const unsigned char *k); + unsigned long long inlen, const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_shorthash_keygen(unsigned char k[crypto_shorthash_KEYBYTES]); +void crypto_shorthash_keygen(unsigned char k[crypto_shorthash_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_shorthash_siphash24.h b/sodium/sodium_crypto_shorthash_siphash24.h similarity index 87% rename from windows/sodium/crypto_shorthash_siphash24.h rename to sodium/sodium_crypto_shorthash_siphash24.h index 745ed48f..cf461a4d 100644 --- a/windows/sodium/crypto_shorthash_siphash24.h +++ b/sodium/sodium_crypto_shorthash_siphash24.h @@ -2,7 +2,7 @@ #define crypto_shorthash_siphash24_H #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -14,31 +14,27 @@ extern "C" { /* -- 64-bit output -- */ #define crypto_shorthash_siphash24_BYTES 8U -SODIUM_EXPORT size_t crypto_shorthash_siphash24_bytes(void); #define crypto_shorthash_siphash24_KEYBYTES 16U -SODIUM_EXPORT size_t crypto_shorthash_siphash24_keybytes(void); -SODIUM_EXPORT int crypto_shorthash_siphash24(unsigned char *out, const unsigned char *in, - unsigned long long inlen, const unsigned char *k); + unsigned long long inlen, const unsigned char *k) + __attribute__ ((nonnull)); #ifndef SODIUM_LIBRARY_MINIMAL /* -- 128-bit output -- */ #define crypto_shorthash_siphashx24_BYTES 16U -SODIUM_EXPORT size_t crypto_shorthash_siphashx24_bytes(void); #define crypto_shorthash_siphashx24_KEYBYTES 16U -SODIUM_EXPORT size_t crypto_shorthash_siphashx24_keybytes(void); -SODIUM_EXPORT int crypto_shorthash_siphashx24(unsigned char *out, const unsigned char *in, - unsigned long long inlen, const unsigned char *k); + unsigned long long inlen, const unsigned char *k) + __attribute__ ((nonnull)); #endif #ifdef __cplusplus diff --git a/sodium/sodium_crypto_sign.c b/sodium/sodium_crypto_sign.c new file mode 100644 index 00000000..8283a57a --- /dev/null +++ b/sodium/sodium_crypto_sign.c @@ -0,0 +1,51 @@ + +#include "sodium_crypto_sign.h" + +int +crypto_sign_keypair(unsigned char *pk, unsigned char *sk) +{ + return crypto_sign_ed25519_keypair(pk, sk); +} + +int +crypto_sign(unsigned char *sm, unsigned long long *smlen_p, + const unsigned char *m, unsigned long long mlen, + const unsigned char *sk) +{ + return crypto_sign_ed25519(sm, smlen_p, m, mlen, sk); +} + +int +crypto_sign_open(unsigned char *m, unsigned long long *mlen_p, + const unsigned char *sm, unsigned long long smlen, + const unsigned char *pk) +{ + return crypto_sign_ed25519_open(m, mlen_p, sm, smlen, pk); +} + +int +crypto_sign_init(crypto_sign_state *state) +{ + return crypto_sign_ed25519ph_init(state); +} + +int +crypto_sign_update(crypto_sign_state *state, const unsigned char *m, + unsigned long long mlen) +{ + return crypto_sign_ed25519ph_update(state, m, mlen); +} + +int +crypto_sign_final_create(crypto_sign_state *state, unsigned char *sig, + unsigned long long *siglen_p, const unsigned char *sk) +{ + return crypto_sign_ed25519ph_final_create(state, sig, siglen_p, sk); +} + +int +crypto_sign_final_verify(crypto_sign_state *state, const unsigned char *sig, + const unsigned char *pk) +{ + return crypto_sign_ed25519ph_final_verify(state, sig, pk); +} diff --git a/windows/sodium/crypto_sign.h b/sodium/sodium_crypto_sign.h similarity index 67% rename from windows/sodium/crypto_sign.h rename to sodium/sodium_crypto_sign.h index b0335bf2..406eba1c 100644 --- a/windows/sodium/crypto_sign.h +++ b/sodium/sodium_crypto_sign.h @@ -10,8 +10,8 @@ #include -#include "crypto_sign_ed25519.h" -#include "export.h" +#include "sodium_crypto_sign_ed25519.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -22,75 +22,66 @@ extern "C" { typedef crypto_sign_ed25519ph_state crypto_sign_state; -SODIUM_EXPORT size_t crypto_sign_statebytes(void); #define crypto_sign_BYTES crypto_sign_ed25519_BYTES -SODIUM_EXPORT size_t crypto_sign_bytes(void); #define crypto_sign_SEEDBYTES crypto_sign_ed25519_SEEDBYTES -SODIUM_EXPORT size_t crypto_sign_seedbytes(void); #define crypto_sign_PUBLICKEYBYTES crypto_sign_ed25519_PUBLICKEYBYTES -SODIUM_EXPORT size_t crypto_sign_publickeybytes(void); #define crypto_sign_SECRETKEYBYTES crypto_sign_ed25519_SECRETKEYBYTES -SODIUM_EXPORT size_t crypto_sign_secretkeybytes(void); +#define crypto_sign_MESSAGEBYTES_MAX crypto_sign_ed25519_MESSAGEBYTES_MAX +size_t crypto_sign_messagebytes_max(void); + #define crypto_sign_PRIMITIVE "ed25519" -SODIUM_EXPORT const char *crypto_sign_primitive(void); -SODIUM_EXPORT int crypto_sign_seed_keypair(unsigned char *pk, unsigned char *sk, - const unsigned char *seed); + const unsigned char *seed) + __attribute__ ((nonnull)); -SODIUM_EXPORT -int crypto_sign_keypair(unsigned char *pk, unsigned char *sk); +int crypto_sign_keypair(unsigned char *pk, unsigned char *sk) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_sign(unsigned char *sm, unsigned long long *smlen_p, const unsigned char *m, unsigned long long mlen, - const unsigned char *sk); + const unsigned char *sk) __attribute__ ((nonnull(1, 3, 5))); -SODIUM_EXPORT int crypto_sign_open(unsigned char *m, unsigned long long *mlen_p, const unsigned char *sm, unsigned long long smlen, const unsigned char *pk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull(3, 5))); -SODIUM_EXPORT int crypto_sign_detached(unsigned char *sig, unsigned long long *siglen_p, const unsigned char *m, unsigned long long mlen, - const unsigned char *sk); + const unsigned char *sk) __attribute__ ((nonnull(1, 3, 5))); -SODIUM_EXPORT int crypto_sign_verify_detached(const unsigned char *sig, const unsigned char *m, unsigned long long mlen, const unsigned char *pk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_sign_init(crypto_sign_state *state); -SODIUM_EXPORT int crypto_sign_update(crypto_sign_state *state, - const unsigned char *m, unsigned long long mlen); + const unsigned char *m, unsigned long long mlen) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_sign_final_create(crypto_sign_state *state, unsigned char *sig, unsigned long long *siglen_p, - const unsigned char *sk); + const unsigned char *sk) + __attribute__ ((nonnull(1, 2, 4))); -SODIUM_EXPORT -int crypto_sign_final_verify(crypto_sign_state *state, unsigned char *sig, +int crypto_sign_final_verify(crypto_sign_state *state, const unsigned char *sig, const unsigned char *pk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_sign_ed25519.h b/sodium/sodium_crypto_sign_ed25519.h similarity index 70% rename from windows/sodium/crypto_sign_ed25519.h rename to sodium/sodium_crypto_sign_ed25519.h index 17c150f2..2f36d701 100644 --- a/windows/sodium/crypto_sign_ed25519.h +++ b/sodium/sodium_crypto_sign_ed25519.h @@ -2,8 +2,8 @@ #define crypto_sign_ed25519_H #include -#include "crypto_hash_sha512.h" -#include "export.h" +#include "sodium_crypto_hash_sha512.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -16,92 +16,84 @@ typedef struct crypto_sign_ed25519ph_state { crypto_hash_sha512_state hs; } crypto_sign_ed25519ph_state; -SODIUM_EXPORT size_t crypto_sign_ed25519ph_statebytes(void); #define crypto_sign_ed25519_BYTES 64U -SODIUM_EXPORT size_t crypto_sign_ed25519_bytes(void); #define crypto_sign_ed25519_SEEDBYTES 32U -SODIUM_EXPORT size_t crypto_sign_ed25519_seedbytes(void); #define crypto_sign_ed25519_PUBLICKEYBYTES 32U -SODIUM_EXPORT size_t crypto_sign_ed25519_publickeybytes(void); #define crypto_sign_ed25519_SECRETKEYBYTES (32U + 32U) -SODIUM_EXPORT size_t crypto_sign_ed25519_secretkeybytes(void); -SODIUM_EXPORT +#define crypto_sign_ed25519_MESSAGEBYTES_MAX (SODIUM_SIZE_MAX - crypto_sign_ed25519_BYTES) +size_t crypto_sign_ed25519_messagebytes_max(void); + int crypto_sign_ed25519(unsigned char *sm, unsigned long long *smlen_p, const unsigned char *m, unsigned long long mlen, - const unsigned char *sk); + const unsigned char *sk) + __attribute__ ((nonnull(1, 3, 5))); -SODIUM_EXPORT int crypto_sign_ed25519_open(unsigned char *m, unsigned long long *mlen_p, const unsigned char *sm, unsigned long long smlen, - const unsigned char *pk) - __attribute__ ((warn_unused_result)); + const unsigned char *pk); -SODIUM_EXPORT int crypto_sign_ed25519_detached(unsigned char *sig, unsigned long long *siglen_p, const unsigned char *m, unsigned long long mlen, - const unsigned char *sk); + const unsigned char *sk) + __attribute__ ((nonnull(1, 3))); -SODIUM_EXPORT int crypto_sign_ed25519_verify_detached(const unsigned char *sig, const unsigned char *m, unsigned long long mlen, const unsigned char *pk) __attribute__ ((warn_unused_result)); -SODIUM_EXPORT -int crypto_sign_ed25519_keypair(unsigned char *pk, unsigned char *sk); +int crypto_sign_ed25519_keypair(unsigned char *pk, unsigned char *sk) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_sign_ed25519_seed_keypair(unsigned char *pk, unsigned char *sk, - const unsigned char *seed); + const unsigned char *seed) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_sign_ed25519_pk_to_curve25519(unsigned char *curve25519_pk, const unsigned char *ed25519_pk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_sign_ed25519_sk_to_curve25519(unsigned char *curve25519_sk, - const unsigned char *ed25519_sk); + const unsigned char *ed25519_sk) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_sign_ed25519_sk_to_seed(unsigned char *seed, - const unsigned char *sk); + const unsigned char *sk) + __attribute__ ((nonnull)); -SODIUM_EXPORT -int crypto_sign_ed25519_sk_to_pk(unsigned char *pk, const unsigned char *sk); +int crypto_sign_ed25519_sk_to_pk(unsigned char *pk, const unsigned char *sk) + __attribute__ ((nonnull)); -SODIUM_EXPORT -int crypto_sign_ed25519ph_init(crypto_sign_ed25519ph_state *state); +int crypto_sign_ed25519ph_init(crypto_sign_ed25519ph_state *state) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_sign_ed25519ph_update(crypto_sign_ed25519ph_state *state, const unsigned char *m, - unsigned long long mlen); + unsigned long long mlen) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_sign_ed25519ph_final_create(crypto_sign_ed25519ph_state *state, unsigned char *sig, unsigned long long *siglen_p, const unsigned char *sk); -SODIUM_EXPORT int crypto_sign_ed25519ph_final_verify(crypto_sign_ed25519ph_state *state, - unsigned char *sig, + const unsigned char *sig, const unsigned char *pk) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_sign_edwards25519sha512batch.h b/sodium/sodium_crypto_sign_edwards25519sha512batch.h similarity index 81% rename from windows/sodium/crypto_sign_edwards25519sha512batch.h rename to sodium/sodium_crypto_sign_edwards25519sha512batch.h index 2224a94e..e9c5bd54 100644 --- a/windows/sodium/crypto_sign_edwards25519sha512batch.h +++ b/sodium/sodium_crypto_sign_edwards25519sha512batch.h @@ -13,7 +13,7 @@ */ #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -25,27 +25,25 @@ extern "C" { #define crypto_sign_edwards25519sha512batch_BYTES 64U #define crypto_sign_edwards25519sha512batch_PUBLICKEYBYTES 32U #define crypto_sign_edwards25519sha512batch_SECRETKEYBYTES (32U + 32U) +#define crypto_sign_edwards25519sha512batch_MESSAGEBYTES_MAX (SODIUM_SIZE_MAX - crypto_sign_edwards25519sha512batch_BYTES) -SODIUM_EXPORT int crypto_sign_edwards25519sha512batch(unsigned char *sm, unsigned long long *smlen_p, const unsigned char *m, unsigned long long mlen, const unsigned char *sk) - __attribute__ ((deprecated)); + __attribute__ ((deprecated)) __attribute__ ((nonnull(1, 3, 5))); -SODIUM_EXPORT int crypto_sign_edwards25519sha512batch_open(unsigned char *m, unsigned long long *mlen_p, const unsigned char *sm, unsigned long long smlen, const unsigned char *pk) - __attribute__ ((deprecated)); + __attribute__ ((deprecated)) __attribute__ ((nonnull(3, 5))); -SODIUM_EXPORT int crypto_sign_edwards25519sha512batch_keypair(unsigned char *pk, unsigned char *sk) - __attribute__ ((deprecated)); + __attribute__ ((deprecated)) __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_stream.h b/sodium/sodium_crypto_stream.h similarity index 76% rename from windows/sodium/crypto_stream.h rename to sodium/sodium_crypto_stream.h index 22de6ff5..5a1c095f 100644 --- a/windows/sodium/crypto_stream.h +++ b/sodium/sodium_crypto_stream.h @@ -11,8 +11,8 @@ #include -#include "crypto_stream_xsalsa20.h" -#include "export.h" +#include "sodium_crypto_stream_xsalsa20.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -22,28 +22,28 @@ extern "C" { #endif #define crypto_stream_KEYBYTES crypto_stream_xsalsa20_KEYBYTES -SODIUM_EXPORT size_t crypto_stream_keybytes(void); #define crypto_stream_NONCEBYTES crypto_stream_xsalsa20_NONCEBYTES -SODIUM_EXPORT size_t crypto_stream_noncebytes(void); +#define crypto_stream_MESSAGEBYTES_MAX crypto_stream_xsalsa20_MESSAGEBYTES_MAX +size_t crypto_stream_messagebytes_max(void); + #define crypto_stream_PRIMITIVE "xsalsa20" -SODIUM_EXPORT const char *crypto_stream_primitive(void); -SODIUM_EXPORT int crypto_stream(unsigned char *c, unsigned long long clen, - const unsigned char *n, const unsigned char *k); + const unsigned char *n, const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_xor(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_stream_keygen(unsigned char k[crypto_stream_KEYBYTES]); +void crypto_stream_keygen(unsigned char k[crypto_stream_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_stream_chacha20.h b/sodium/sodium_crypto_stream_chacha20.h similarity index 70% rename from windows/sodium/crypto_stream_chacha20.h rename to sodium/sodium_crypto_stream_chacha20.h index 352b9290..ed20fbe8 100644 --- a/windows/sodium/crypto_stream_chacha20.h +++ b/sodium/sodium_crypto_stream_chacha20.h @@ -11,7 +11,7 @@ #include #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -21,65 +21,69 @@ extern "C" { #endif #define crypto_stream_chacha20_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_stream_chacha20_keybytes(void); #define crypto_stream_chacha20_NONCEBYTES 8U -SODIUM_EXPORT size_t crypto_stream_chacha20_noncebytes(void); +#define crypto_stream_chacha20_MESSAGEBYTES_MAX SODIUM_SIZE_MAX +size_t crypto_stream_chacha20_messagebytes_max(void); + /* ChaCha20 with a 64-bit nonce and a 64-bit counter, as originally designed */ -SODIUM_EXPORT int crypto_stream_chacha20(unsigned char *c, unsigned long long clen, - const unsigned char *n, const unsigned char *k); + const unsigned char *n, const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_chacha20_xor(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_chacha20_xor_ic(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, uint64_t ic, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_stream_chacha20_keygen(unsigned char k[crypto_stream_chacha20_KEYBYTES]); +void crypto_stream_chacha20_keygen(unsigned char k[crypto_stream_chacha20_KEYBYTES]) + __attribute__ ((nonnull)); /* ChaCha20 with a 96-bit nonce and a 32-bit counter (IETF) */ #define crypto_stream_chacha20_ietf_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_stream_chacha20_ietf_keybytes(void); #define crypto_stream_chacha20_ietf_NONCEBYTES 12U -SODIUM_EXPORT size_t crypto_stream_chacha20_ietf_noncebytes(void); -SODIUM_EXPORT +#define crypto_stream_chacha20_ietf_MESSAGEBYTES_MAX \ + SODIUM_MIN(SODIUM_SIZE_MAX, 64ULL * (1ULL << 32)) +size_t crypto_stream_chacha20_ietf_messagebytes_max(void); + int crypto_stream_chacha20_ietf(unsigned char *c, unsigned long long clen, - const unsigned char *n, const unsigned char *k); + const unsigned char *n, const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_chacha20_ietf_xor(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_chacha20_ietf_xor_ic(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, uint32_t ic, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_stream_chacha20_ietf_keygen(unsigned char k[crypto_stream_chacha20_ietf_KEYBYTES]); +void crypto_stream_chacha20_ietf_keygen(unsigned char k[crypto_stream_chacha20_ietf_KEYBYTES]) + __attribute__ ((nonnull)); /* Aliases */ #define crypto_stream_chacha20_IETF_KEYBYTES crypto_stream_chacha20_ietf_KEYBYTES #define crypto_stream_chacha20_IETF_NONCEBYTES crypto_stream_chacha20_ietf_NONCEBYTES +#define crypto_stream_chacha20_IETF_MESSAGEBYTES_MAX crypto_stream_chacha20_ietf_MESSAGEBYTES_MAX #ifdef __cplusplus } diff --git a/windows/sodium/crypto_stream_salsa20.h b/sodium/sodium_crypto_stream_salsa20.h similarity index 74% rename from windows/sodium/crypto_stream_salsa20.h rename to sodium/sodium_crypto_stream_salsa20.h index 961e5c1c..80aee376 100644 --- a/windows/sodium/crypto_stream_salsa20.h +++ b/sodium/sodium_crypto_stream_salsa20.h @@ -11,7 +11,7 @@ #include #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -21,30 +21,31 @@ extern "C" { #endif #define crypto_stream_salsa20_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_stream_salsa20_keybytes(void); #define crypto_stream_salsa20_NONCEBYTES 8U -SODIUM_EXPORT size_t crypto_stream_salsa20_noncebytes(void); -SODIUM_EXPORT +#define crypto_stream_salsa20_MESSAGEBYTES_MAX SODIUM_SIZE_MAX +size_t crypto_stream_salsa20_messagebytes_max(void); + int crypto_stream_salsa20(unsigned char *c, unsigned long long clen, - const unsigned char *n, const unsigned char *k); + const unsigned char *n, const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_salsa20_xor(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_salsa20_xor_ic(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, uint64_t ic, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_stream_salsa20_keygen(unsigned char k[crypto_stream_salsa20_KEYBYTES]); +void crypto_stream_salsa20_keygen(unsigned char k[crypto_stream_salsa20_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_stream_salsa2012.h b/sodium/sodium_crypto_stream_salsa2012.h similarity index 75% rename from windows/sodium/crypto_stream_salsa2012.h rename to sodium/sodium_crypto_stream_salsa2012.h index d5c44282..b3fecdfd 100644 --- a/windows/sodium/crypto_stream_salsa2012.h +++ b/sodium/sodium_crypto_stream_salsa2012.h @@ -10,7 +10,7 @@ */ #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -20,24 +20,25 @@ extern "C" { #endif #define crypto_stream_salsa2012_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_stream_salsa2012_keybytes(void); #define crypto_stream_salsa2012_NONCEBYTES 8U -SODIUM_EXPORT size_t crypto_stream_salsa2012_noncebytes(void); -SODIUM_EXPORT +#define crypto_stream_salsa2012_MESSAGEBYTES_MAX SODIUM_SIZE_MAX +size_t crypto_stream_salsa2012_messagebytes_max(void); + int crypto_stream_salsa2012(unsigned char *c, unsigned long long clen, - const unsigned char *n, const unsigned char *k); + const unsigned char *n, const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_salsa2012_xor(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_stream_salsa2012_keygen(unsigned char k[crypto_stream_salsa2012_KEYBYTES]); +void crypto_stream_salsa2012_keygen(unsigned char k[crypto_stream_salsa2012_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_stream_salsa208.h b/sodium/sodium_crypto_stream_salsa208.h similarity index 60% rename from windows/sodium/crypto_stream_salsa208.h rename to sodium/sodium_crypto_stream_salsa208.h index 02b4166e..e5d2a7d0 100644 --- a/windows/sodium/crypto_stream_salsa208.h +++ b/sodium/sodium_crypto_stream_salsa208.h @@ -10,7 +10,7 @@ */ #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -20,24 +20,28 @@ extern "C" { #endif #define crypto_stream_salsa208_KEYBYTES 32U -SODIUM_EXPORT -size_t crypto_stream_salsa208_keybytes(void); +size_t crypto_stream_salsa208_keybytes(void) + __attribute__ ((deprecated)); #define crypto_stream_salsa208_NONCEBYTES 8U -SODIUM_EXPORT -size_t crypto_stream_salsa208_noncebytes(void); +size_t crypto_stream_salsa208_noncebytes(void) + __attribute__ ((deprecated)); + +#define crypto_stream_salsa208_MESSAGEBYTES_MAX SODIUM_SIZE_MAX +size_t crypto_stream_salsa208_messagebytes_max(void) + __attribute__ ((deprecated)); -SODIUM_EXPORT int crypto_stream_salsa208(unsigned char *c, unsigned long long clen, - const unsigned char *n, const unsigned char *k); + const unsigned char *n, const unsigned char *k) + __attribute__ ((deprecated)) __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_salsa208_xor(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((deprecated)) __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_stream_salsa208_keygen(unsigned char k[crypto_stream_salsa208_KEYBYTES]); +void crypto_stream_salsa208_keygen(unsigned char k[crypto_stream_salsa208_KEYBYTES]) + __attribute__ ((deprecated)) __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_stream_xchacha20.h b/sodium/sodium_crypto_stream_xchacha20.h similarity index 74% rename from windows/sodium/crypto_stream_xchacha20.h rename to sodium/sodium_crypto_stream_xchacha20.h index f884798e..0a5e0aea 100644 --- a/windows/sodium/crypto_stream_xchacha20.h +++ b/sodium/sodium_crypto_stream_xchacha20.h @@ -11,7 +11,7 @@ #include #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -21,30 +21,31 @@ extern "C" { #endif #define crypto_stream_xchacha20_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_stream_xchacha20_keybytes(void); #define crypto_stream_xchacha20_NONCEBYTES 24U -SODIUM_EXPORT size_t crypto_stream_xchacha20_noncebytes(void); -SODIUM_EXPORT +#define crypto_stream_xchacha20_MESSAGEBYTES_MAX SODIUM_SIZE_MAX +size_t crypto_stream_xchacha20_messagebytes_max(void); + int crypto_stream_xchacha20(unsigned char *c, unsigned long long clen, - const unsigned char *n, const unsigned char *k); + const unsigned char *n, const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_xchacha20_xor(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_xchacha20_xor_ic(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, uint64_t ic, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_stream_xchacha20_keygen(unsigned char k[crypto_stream_xchacha20_KEYBYTES]); +void crypto_stream_xchacha20_keygen(unsigned char k[crypto_stream_xchacha20_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_stream_xsalsa20.h b/sodium/sodium_crypto_stream_xsalsa20.h similarity index 74% rename from windows/sodium/crypto_stream_xsalsa20.h rename to sodium/sodium_crypto_stream_xsalsa20.h index ed5ae3c3..3c49449d 100644 --- a/windows/sodium/crypto_stream_xsalsa20.h +++ b/sodium/sodium_crypto_stream_xsalsa20.h @@ -11,7 +11,7 @@ #include #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -21,30 +21,31 @@ extern "C" { #endif #define crypto_stream_xsalsa20_KEYBYTES 32U -SODIUM_EXPORT size_t crypto_stream_xsalsa20_keybytes(void); #define crypto_stream_xsalsa20_NONCEBYTES 24U -SODIUM_EXPORT size_t crypto_stream_xsalsa20_noncebytes(void); -SODIUM_EXPORT +#define crypto_stream_xsalsa20_MESSAGEBYTES_MAX SODIUM_SIZE_MAX +size_t crypto_stream_xsalsa20_messagebytes_max(void); + int crypto_stream_xsalsa20(unsigned char *c, unsigned long long clen, - const unsigned char *n, const unsigned char *k); + const unsigned char *n, const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_xsalsa20_xor(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT int crypto_stream_xsalsa20_xor_ic(unsigned char *c, const unsigned char *m, unsigned long long mlen, const unsigned char *n, uint64_t ic, - const unsigned char *k); + const unsigned char *k) + __attribute__ ((nonnull)); -SODIUM_EXPORT -void crypto_stream_xsalsa20_keygen(unsigned char k[crypto_stream_xsalsa20_KEYBYTES]); +void crypto_stream_xsalsa20_keygen(unsigned char k[crypto_stream_xsalsa20_KEYBYTES]) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_verify_16.h b/sodium/sodium_crypto_verify_16.h similarity index 73% rename from windows/sodium/crypto_verify_16.h rename to sodium/sodium_crypto_verify_16.h index 5e9eeabe..94e04c4c 100644 --- a/windows/sodium/crypto_verify_16.h +++ b/sodium/sodium_crypto_verify_16.h @@ -2,19 +2,18 @@ #define crypto_verify_16_H #include -#include "export.h" + +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif #define crypto_verify_16_BYTES 16U -SODIUM_EXPORT size_t crypto_verify_16_bytes(void); -SODIUM_EXPORT int crypto_verify_16(const unsigned char *x, const unsigned char *y) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_verify_32.h b/sodium/sodium_crypto_verify_32.h similarity index 74% rename from windows/sodium/crypto_verify_32.h rename to sodium/sodium_crypto_verify_32.h index 281b5a1b..8f9b00b3 100644 --- a/windows/sodium/crypto_verify_32.h +++ b/sodium/sodium_crypto_verify_32.h @@ -2,19 +2,17 @@ #define crypto_verify_32_H #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif #define crypto_verify_32_BYTES 32U -SODIUM_EXPORT size_t crypto_verify_32_bytes(void); -SODIUM_EXPORT int crypto_verify_32(const unsigned char *x, const unsigned char *y) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/windows/sodium/crypto_verify_64.h b/sodium/sodium_crypto_verify_64.h similarity index 74% rename from windows/sodium/crypto_verify_64.h rename to sodium/sodium_crypto_verify_64.h index 0dc7c304..e0b7bc94 100644 --- a/windows/sodium/crypto_verify_64.h +++ b/sodium/sodium_crypto_verify_64.h @@ -2,19 +2,17 @@ #define crypto_verify_64_H #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif #define crypto_verify_64_BYTES 64U -SODIUM_EXPORT size_t crypto_verify_64_bytes(void); -SODIUM_EXPORT int crypto_verify_64(const unsigned char *x, const unsigned char *y) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/sodium/sodium_dolbeau_chacha20-avx2.c b/sodium/sodium_dolbeau_chacha20-avx2.c new file mode 100644 index 00000000..9eef398e --- /dev/null +++ b/sodium/sodium_dolbeau_chacha20-avx2.c @@ -0,0 +1,179 @@ + +#include +#include +#include + +#include "sodium_core.h" +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_private_common.h" +#include "sodium_private_sse2_64_32.h" +#include "sodium_utils.h" + +#if defined(HAVE_AVX2INTRIN_H) && defined(HAVE_EMMINTRIN_H) && \ + defined(HAVE_TMMINTRIN_H) && defined(HAVE_SMMINTRIN_H) + +# ifdef __GNUC__ +# pragma GCC target("sse2") +# pragma GCC target("ssse3") +# pragma GCC target("sse4.1") +# pragma GCC target("avx2") +# endif + +# include +# include +# include +# include + +# include "sodium_stream_chacha20.h" +# include "sodium_dolbeau_chacha20-avx2.h" + +# define ROUNDS 20 + +typedef struct chacha_ctx { + uint32_t input[16]; +} chacha_ctx; + +static void +chacha_keysetup(chacha_ctx *ctx, const uint8_t *k) +{ + ctx->input[0] = 0x61707865; + ctx->input[1] = 0x3320646e; + ctx->input[2] = 0x79622d32; + ctx->input[3] = 0x6b206574; + ctx->input[4] = LOAD32_LE(k + 0); + ctx->input[5] = LOAD32_LE(k + 4); + ctx->input[6] = LOAD32_LE(k + 8); + ctx->input[7] = LOAD32_LE(k + 12); + ctx->input[8] = LOAD32_LE(k + 16); + ctx->input[9] = LOAD32_LE(k + 20); + ctx->input[10] = LOAD32_LE(k + 24); + ctx->input[11] = LOAD32_LE(k + 28); +} + +static void +chacha_ivsetup(chacha_ctx *ctx, const uint8_t *iv, const uint8_t *counter) +{ + ctx->input[12] = counter == NULL ? 0 : LOAD32_LE(counter + 0); + ctx->input[13] = counter == NULL ? 0 : LOAD32_LE(counter + 4); + ctx->input[14] = LOAD32_LE(iv + 0); + ctx->input[15] = LOAD32_LE(iv + 4); +} + +static void +chacha_ietf_ivsetup(chacha_ctx *ctx, const uint8_t *iv, const uint8_t *counter) +{ + ctx->input[12] = counter == NULL ? 0 : LOAD32_LE(counter); + ctx->input[13] = LOAD32_LE(iv + 0); + ctx->input[14] = LOAD32_LE(iv + 4); + ctx->input[15] = LOAD32_LE(iv + 8); +} + +static void +chacha20_encrypt_bytes(chacha_ctx *ctx, const uint8_t *m, uint8_t *c, + unsigned long long bytes) +{ + uint32_t * const x = &ctx->input[0]; + + if (!bytes) { + return; /* LCOV_EXCL_LINE */ + } +# include "sodium_dolbeau_u8.h" +# include "sodium_dolbeau_u4.h" +# include "sodium_dolbeau_u1.h" +# include "sodium_dolbeau_u0.h" +} + +static int +stream_ref(unsigned char *c, unsigned long long clen, const unsigned char *n, + const unsigned char *k) +{ + struct chacha_ctx ctx; + + if (!clen) { + return 0; + } + COMPILER_ASSERT(crypto_stream_chacha20_KEYBYTES == 256 / 8); + chacha_keysetup(&ctx, k); + chacha_ivsetup(&ctx, n, NULL); + memset(c, 0, clen); + chacha20_encrypt_bytes(&ctx, c, c, clen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +static int +stream_ietf_ext_ref(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k) +{ + struct chacha_ctx ctx; + + if (!clen) { + return 0; + } + COMPILER_ASSERT(crypto_stream_chacha20_KEYBYTES == 256 / 8); + chacha_keysetup(&ctx, k); + chacha_ietf_ivsetup(&ctx, n, NULL); + memset(c, 0, clen); + chacha20_encrypt_bytes(&ctx, c, c, clen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +static int +stream_ref_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, uint64_t ic, + const unsigned char *k) +{ + struct chacha_ctx ctx; + uint8_t ic_bytes[8]; + uint32_t ic_high; + uint32_t ic_low; + + if (!mlen) { + return 0; + } + ic_high = (uint32_t) (ic >> 32); + ic_low = (uint32_t) ic; + STORE32_LE(&ic_bytes[0], ic_low); + STORE32_LE(&ic_bytes[4], ic_high); + chacha_keysetup(&ctx, k); + chacha_ivsetup(&ctx, n, ic_bytes); + chacha20_encrypt_bytes(&ctx, m, c, mlen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +static int +stream_ietf_ext_ref_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + uint32_t ic, const unsigned char *k) +{ + struct chacha_ctx ctx; + uint8_t ic_bytes[4]; + + if (!mlen) { + return 0; + } + STORE32_LE(ic_bytes, ic); + chacha_keysetup(&ctx, k); + chacha_ietf_ivsetup(&ctx, n, ic_bytes); + chacha20_encrypt_bytes(&ctx, m, c, mlen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +struct crypto_stream_chacha20_implementation + crypto_stream_chacha20_dolbeau_avx2_implementation = { + SODIUM_C99(.stream =) stream_ref, + SODIUM_C99(.stream_ietf_ext =) stream_ietf_ext_ref, + SODIUM_C99(.stream_xor_ic =) stream_ref_xor_ic, + SODIUM_C99(.stream_ietf_ext_xor_ic =) stream_ietf_ext_ref_xor_ic + }; + +#endif + +int chacha20_dolbeau_link_warning_dummy = 0; diff --git a/sodium/sodium_dolbeau_chacha20-avx2.h b/sodium/sodium_dolbeau_chacha20-avx2.h new file mode 100644 index 00000000..2c47ab44 --- /dev/null +++ b/sodium/sodium_dolbeau_chacha20-avx2.h @@ -0,0 +1,8 @@ + +#include + +#include "sodium_stream_chacha20.h" +#include "sodium_crypto_stream_chacha20.h" + +extern struct crypto_stream_chacha20_implementation + crypto_stream_chacha20_dolbeau_avx2_implementation; diff --git a/sodium/sodium_dolbeau_chacha20-ssse3.c b/sodium/sodium_dolbeau_chacha20-ssse3.c new file mode 100644 index 00000000..ad9c3256 --- /dev/null +++ b/sodium/sodium_dolbeau_chacha20-ssse3.c @@ -0,0 +1,173 @@ + +#include +#include +#include + +#include "sodium_core.h" +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_private_common.h" +#include "sodium_private_sse2_64_32.h" +#include "sodium_utils.h" + +#if defined(HAVE_EMMINTRIN_H) && defined(HAVE_TMMINTRIN_H) + +# ifdef __GNUC__ +# pragma GCC target("sse2") +# pragma GCC target("ssse3") +# endif + +# include +# include + +# include "sodium_stream_chacha20.h" +# include "sodium_dolbeau_chacha20-ssse3.h" + +# define ROUNDS 20 + +typedef struct chacha_ctx { + uint32_t input[16]; +} chacha_ctx; + +static void +chacha_keysetup(chacha_ctx *ctx, const uint8_t *k) +{ + ctx->input[0] = 0x61707865; + ctx->input[1] = 0x3320646e; + ctx->input[2] = 0x79622d32; + ctx->input[3] = 0x6b206574; + ctx->input[4] = LOAD32_LE(k + 0); + ctx->input[5] = LOAD32_LE(k + 4); + ctx->input[6] = LOAD32_LE(k + 8); + ctx->input[7] = LOAD32_LE(k + 12); + ctx->input[8] = LOAD32_LE(k + 16); + ctx->input[9] = LOAD32_LE(k + 20); + ctx->input[10] = LOAD32_LE(k + 24); + ctx->input[11] = LOAD32_LE(k + 28); +} + +static void +chacha_ivsetup(chacha_ctx *ctx, const uint8_t *iv, const uint8_t *counter) +{ + ctx->input[12] = counter == NULL ? 0 : LOAD32_LE(counter + 0); + ctx->input[13] = counter == NULL ? 0 : LOAD32_LE(counter + 4); + ctx->input[14] = LOAD32_LE(iv + 0); + ctx->input[15] = LOAD32_LE(iv + 4); +} + +static void +chacha_ietf_ivsetup(chacha_ctx *ctx, const uint8_t *iv, const uint8_t *counter) +{ + ctx->input[12] = counter == NULL ? 0 : LOAD32_LE(counter); + ctx->input[13] = LOAD32_LE(iv + 0); + ctx->input[14] = LOAD32_LE(iv + 4); + ctx->input[15] = LOAD32_LE(iv + 8); +} + +static void +chacha20_encrypt_bytes(chacha_ctx *ctx, const uint8_t *m, uint8_t *c, + unsigned long long bytes) +{ + uint32_t * const x = &ctx->input[0]; + + if (!bytes) { + return; /* LCOV_EXCL_LINE */ + } +# include "sodium_dolbeau_u4.h" +# include "sodium_dolbeau_u1.h" +# include "sodium_dolbeau_u0.h" +} + +static int +stream_ref(unsigned char *c, unsigned long long clen, const unsigned char *n, + const unsigned char *k) +{ + struct chacha_ctx ctx; + + if (!clen) { + return 0; + } + COMPILER_ASSERT(crypto_stream_chacha20_KEYBYTES == 256 / 8); + chacha_keysetup(&ctx, k); + chacha_ivsetup(&ctx, n, NULL); + memset(c, 0, clen); + chacha20_encrypt_bytes(&ctx, c, c, clen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +static int +stream_ietf_ext_ref(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k) +{ + struct chacha_ctx ctx; + + if (!clen) { + return 0; + } + COMPILER_ASSERT(crypto_stream_chacha20_KEYBYTES == 256 / 8); + chacha_keysetup(&ctx, k); + chacha_ietf_ivsetup(&ctx, n, NULL); + memset(c, 0, clen); + chacha20_encrypt_bytes(&ctx, c, c, clen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +static int +stream_ref_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, uint64_t ic, + const unsigned char *k) +{ + struct chacha_ctx ctx; + uint8_t ic_bytes[8]; + uint32_t ic_high; + uint32_t ic_low; + + if (!mlen) { + return 0; + } + ic_high = (uint32_t) (ic >> 32); + ic_low = (uint32_t) ic; + STORE32_LE(&ic_bytes[0], ic_low); + STORE32_LE(&ic_bytes[4], ic_high); + chacha_keysetup(&ctx, k); + chacha_ivsetup(&ctx, n, ic_bytes); + chacha20_encrypt_bytes(&ctx, m, c, mlen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +static int +stream_ietf_ext_ref_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + uint32_t ic, const unsigned char *k) +{ + struct chacha_ctx ctx; + uint8_t ic_bytes[4]; + + if (!mlen) { + return 0; + } + STORE32_LE(ic_bytes, ic); + chacha_keysetup(&ctx, k); + chacha_ietf_ivsetup(&ctx, n, ic_bytes); + chacha20_encrypt_bytes(&ctx, m, c, mlen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +struct crypto_stream_chacha20_implementation + crypto_stream_chacha20_dolbeau_ssse3_implementation = { + SODIUM_C99(.stream =) stream_ref, + SODIUM_C99(.stream_ietf_ext =) stream_ietf_ext_ref, + SODIUM_C99(.stream_xor_ic =) stream_ref_xor_ic, + SODIUM_C99(.stream_ietf_ext_xor_ic =) stream_ietf_ext_ref_xor_ic + }; + +#endif + +int chacha20_dolbeau_ssse3_link_warning_dummy = 0; diff --git a/sodium/sodium_dolbeau_chacha20-ssse3.h b/sodium/sodium_dolbeau_chacha20-ssse3.h new file mode 100644 index 00000000..0f33446c --- /dev/null +++ b/sodium/sodium_dolbeau_chacha20-ssse3.h @@ -0,0 +1,8 @@ + +#include + +#include "sodium_stream_chacha20.h" +#include "sodium_crypto_stream_chacha20.h" + +extern struct crypto_stream_chacha20_implementation + crypto_stream_chacha20_dolbeau_ssse3_implementation; diff --git a/sodium/sodium_dolbeau_u0.h b/sodium/sodium_dolbeau_u0.h new file mode 100644 index 00000000..c05dfd72 --- /dev/null +++ b/sodium/sodium_dolbeau_u0.h @@ -0,0 +1,86 @@ +if (bytes > 0) { + __m128i x_0, x_1, x_2, x_3; + __m128i t_1; + const __m128i rot16 = + _mm_set_epi8(13, 12, 15, 14, 9, 8, 11, 10, 5, 4, 7, 6, 1, 0, 3, 2); + const __m128i rot8 = + _mm_set_epi8(14, 13, 12, 15, 10, 9, 8, 11, 6, 5, 4, 7, 2, 1, 0, 3); + uint8_t partialblock[64]; + + unsigned int i; + + x_0 = _mm_loadu_si128((const __m128i*) (x + 0)); + x_1 = _mm_loadu_si128((const __m128i*) (x + 4)); + x_2 = _mm_loadu_si128((const __m128i*) (x + 8)); + x_3 = _mm_loadu_si128((const __m128i*) (x + 12)); + + for (i = 0; i < ROUNDS; i += 2) { + x_0 = _mm_add_epi32(x_0, x_1); + x_3 = _mm_xor_si128(x_3, x_0); + x_3 = _mm_shuffle_epi8(x_3, rot16); + + x_2 = _mm_add_epi32(x_2, x_3); + x_1 = _mm_xor_si128(x_1, x_2); + + t_1 = x_1; + x_1 = _mm_slli_epi32(x_1, 12); + t_1 = _mm_srli_epi32(t_1, 20); + x_1 = _mm_xor_si128(x_1, t_1); + + x_0 = _mm_add_epi32(x_0, x_1); + x_3 = _mm_xor_si128(x_3, x_0); + x_0 = _mm_shuffle_epi32(x_0, 0x93); + x_3 = _mm_shuffle_epi8(x_3, rot8); + + x_2 = _mm_add_epi32(x_2, x_3); + x_3 = _mm_shuffle_epi32(x_3, 0x4e); + x_1 = _mm_xor_si128(x_1, x_2); + x_2 = _mm_shuffle_epi32(x_2, 0x39); + + t_1 = x_1; + x_1 = _mm_slli_epi32(x_1, 7); + t_1 = _mm_srli_epi32(t_1, 25); + x_1 = _mm_xor_si128(x_1, t_1); + + x_0 = _mm_add_epi32(x_0, x_1); + x_3 = _mm_xor_si128(x_3, x_0); + x_3 = _mm_shuffle_epi8(x_3, rot16); + + x_2 = _mm_add_epi32(x_2, x_3); + x_1 = _mm_xor_si128(x_1, x_2); + + t_1 = x_1; + x_1 = _mm_slli_epi32(x_1, 12); + t_1 = _mm_srli_epi32(t_1, 20); + x_1 = _mm_xor_si128(x_1, t_1); + + x_0 = _mm_add_epi32(x_0, x_1); + x_3 = _mm_xor_si128(x_3, x_0); + x_0 = _mm_shuffle_epi32(x_0, 0x39); + x_3 = _mm_shuffle_epi8(x_3, rot8); + + x_2 = _mm_add_epi32(x_2, x_3); + x_3 = _mm_shuffle_epi32(x_3, 0x4e); + x_1 = _mm_xor_si128(x_1, x_2); + x_2 = _mm_shuffle_epi32(x_2, 0x93); + + t_1 = x_1; + x_1 = _mm_slli_epi32(x_1, 7); + t_1 = _mm_srli_epi32(t_1, 25); + x_1 = _mm_xor_si128(x_1, t_1); + } + x_0 = _mm_add_epi32(x_0, _mm_loadu_si128((const __m128i*) (x + 0))); + x_1 = _mm_add_epi32(x_1, _mm_loadu_si128((const __m128i*) (x + 4))); + x_2 = _mm_add_epi32(x_2, _mm_loadu_si128((const __m128i*) (x + 8))); + x_3 = _mm_add_epi32(x_3, _mm_loadu_si128((const __m128i*) (x + 12))); + _mm_storeu_si128((__m128i*) (partialblock + 0), x_0); + _mm_storeu_si128((__m128i*) (partialblock + 16), x_1); + _mm_storeu_si128((__m128i*) (partialblock + 32), x_2); + _mm_storeu_si128((__m128i*) (partialblock + 48), x_3); + + for (i = 0; i < bytes; i++) { + c[i] = m[i] ^ partialblock[i]; + } + + sodium_memzero(partialblock, sizeof partialblock); +} diff --git a/sodium/sodium_dolbeau_u1.h b/sodium/sodium_dolbeau_u1.h new file mode 100644 index 00000000..f93fffea --- /dev/null +++ b/sodium/sodium_dolbeau_u1.h @@ -0,0 +1,98 @@ +while (bytes >= 64) { + __m128i x_0, x_1, x_2, x_3; + __m128i t_1; + const __m128i rot16 = + _mm_set_epi8(13, 12, 15, 14, 9, 8, 11, 10, 5, 4, 7, 6, 1, 0, 3, 2); + const __m128i rot8 = + _mm_set_epi8(14, 13, 12, 15, 10, 9, 8, 11, 6, 5, 4, 7, 2, 1, 0, 3); + + uint32_t in12; + uint32_t in13; + int i; + + x_0 = _mm_loadu_si128((const __m128i*) (x + 0)); + x_1 = _mm_loadu_si128((const __m128i*) (x + 4)); + x_2 = _mm_loadu_si128((const __m128i*) (x + 8)); + x_3 = _mm_loadu_si128((const __m128i*) (x + 12)); + + for (i = 0; i < ROUNDS; i += 2) { + x_0 = _mm_add_epi32(x_0, x_1); + x_3 = _mm_xor_si128(x_3, x_0); + x_3 = _mm_shuffle_epi8(x_3, rot16); + + x_2 = _mm_add_epi32(x_2, x_3); + x_1 = _mm_xor_si128(x_1, x_2); + + t_1 = x_1; + x_1 = _mm_slli_epi32(x_1, 12); + t_1 = _mm_srli_epi32(t_1, 20); + x_1 = _mm_xor_si128(x_1, t_1); + + x_0 = _mm_add_epi32(x_0, x_1); + x_3 = _mm_xor_si128(x_3, x_0); + x_0 = _mm_shuffle_epi32(x_0, 0x93); + x_3 = _mm_shuffle_epi8(x_3, rot8); + + x_2 = _mm_add_epi32(x_2, x_3); + x_3 = _mm_shuffle_epi32(x_3, 0x4e); + x_1 = _mm_xor_si128(x_1, x_2); + x_2 = _mm_shuffle_epi32(x_2, 0x39); + + t_1 = x_1; + x_1 = _mm_slli_epi32(x_1, 7); + t_1 = _mm_srli_epi32(t_1, 25); + x_1 = _mm_xor_si128(x_1, t_1); + + x_0 = _mm_add_epi32(x_0, x_1); + x_3 = _mm_xor_si128(x_3, x_0); + x_3 = _mm_shuffle_epi8(x_3, rot16); + + x_2 = _mm_add_epi32(x_2, x_3); + x_1 = _mm_xor_si128(x_1, x_2); + + t_1 = x_1; + x_1 = _mm_slli_epi32(x_1, 12); + t_1 = _mm_srli_epi32(t_1, 20); + x_1 = _mm_xor_si128(x_1, t_1); + + x_0 = _mm_add_epi32(x_0, x_1); + x_3 = _mm_xor_si128(x_3, x_0); + x_0 = _mm_shuffle_epi32(x_0, 0x39); + x_3 = _mm_shuffle_epi8(x_3, rot8); + + x_2 = _mm_add_epi32(x_2, x_3); + x_3 = _mm_shuffle_epi32(x_3, 0x4e); + x_1 = _mm_xor_si128(x_1, x_2); + x_2 = _mm_shuffle_epi32(x_2, 0x93); + + t_1 = x_1; + x_1 = _mm_slli_epi32(x_1, 7); + t_1 = _mm_srli_epi32(t_1, 25); + x_1 = _mm_xor_si128(x_1, t_1); + } + x_0 = _mm_add_epi32(x_0, _mm_loadu_si128((const __m128i*) (x + 0))); + x_1 = _mm_add_epi32(x_1, _mm_loadu_si128((const __m128i*) (x + 4))); + x_2 = _mm_add_epi32(x_2, _mm_loadu_si128((const __m128i*) (x + 8))); + x_3 = _mm_add_epi32(x_3, _mm_loadu_si128((const __m128i*) (x + 12))); + x_0 = _mm_xor_si128(x_0, _mm_loadu_si128((const __m128i*) (m + 0))); + x_1 = _mm_xor_si128(x_1, _mm_loadu_si128((const __m128i*) (m + 16))); + x_2 = _mm_xor_si128(x_2, _mm_loadu_si128((const __m128i*) (m + 32))); + x_3 = _mm_xor_si128(x_3, _mm_loadu_si128((const __m128i*) (m + 48))); + _mm_storeu_si128((__m128i*) (c + 0), x_0); + _mm_storeu_si128((__m128i*) (c + 16), x_1); + _mm_storeu_si128((__m128i*) (c + 32), x_2); + _mm_storeu_si128((__m128i*) (c + 48), x_3); + + in12 = x[12]; + in13 = x[13]; + in12++; + if (in12 == 0) { + in13++; + } + x[12] = in12; + x[13] = in13; + + bytes -= 64; + c += 64; + m += 64; +} diff --git a/sodium/sodium_dolbeau_u4.h b/sodium/sodium_dolbeau_u4.h new file mode 100644 index 00000000..4ab295d7 --- /dev/null +++ b/sodium/sodium_dolbeau_u4.h @@ -0,0 +1,177 @@ + +#define VEC4_ROT(A, IMM) \ + _mm_or_si128(_mm_slli_epi32(A, IMM), _mm_srli_epi32(A, (32 - IMM))) + +/* same, but replace 2 of the shift/shift/or "rotation" by byte shuffles (8 & + * 16) (better) */ +#define VEC4_QUARTERROUND_SHUFFLE(A, B, C, D) \ + x_##A = _mm_add_epi32(x_##A, x_##B); \ + t_##A = _mm_xor_si128(x_##D, x_##A); \ + x_##D = _mm_shuffle_epi8(t_##A, rot16); \ + x_##C = _mm_add_epi32(x_##C, x_##D); \ + t_##C = _mm_xor_si128(x_##B, x_##C); \ + x_##B = VEC4_ROT(t_##C, 12); \ + x_##A = _mm_add_epi32(x_##A, x_##B); \ + t_##A = _mm_xor_si128(x_##D, x_##A); \ + x_##D = _mm_shuffle_epi8(t_##A, rot8); \ + x_##C = _mm_add_epi32(x_##C, x_##D); \ + t_##C = _mm_xor_si128(x_##B, x_##C); \ + x_##B = VEC4_ROT(t_##C, 7) + +#define VEC4_QUARTERROUND(A, B, C, D) VEC4_QUARTERROUND_SHUFFLE(A, B, C, D) + +if (bytes >= 256) { + /* constant for shuffling bytes (replacing multiple-of-8 rotates) */ + __m128i rot16 = + _mm_set_epi8(13, 12, 15, 14, 9, 8, 11, 10, 5, 4, 7, 6, 1, 0, 3, 2); + __m128i rot8 = + _mm_set_epi8(14, 13, 12, 15, 10, 9, 8, 11, 6, 5, 4, 7, 2, 1, 0, 3); + + __m128i x_0 = _mm_set1_epi32(x[0]); + __m128i x_1 = _mm_set1_epi32(x[1]); + __m128i x_2 = _mm_set1_epi32(x[2]); + __m128i x_3 = _mm_set1_epi32(x[3]); + __m128i x_4 = _mm_set1_epi32(x[4]); + __m128i x_5 = _mm_set1_epi32(x[5]); + __m128i x_6 = _mm_set1_epi32(x[6]); + __m128i x_7 = _mm_set1_epi32(x[7]); + __m128i x_8 = _mm_set1_epi32(x[8]); + __m128i x_9 = _mm_set1_epi32(x[9]); + __m128i x_10 = _mm_set1_epi32(x[10]); + __m128i x_11 = _mm_set1_epi32(x[11]); + __m128i x_12; + __m128i x_13; + __m128i x_14 = _mm_set1_epi32(x[14]); + __m128i x_15 = _mm_set1_epi32(x[15]); + __m128i orig0 = x_0; + __m128i orig1 = x_1; + __m128i orig2 = x_2; + __m128i orig3 = x_3; + __m128i orig4 = x_4; + __m128i orig5 = x_5; + __m128i orig6 = x_6; + __m128i orig7 = x_7; + __m128i orig8 = x_8; + __m128i orig9 = x_9; + __m128i orig10 = x_10; + __m128i orig11 = x_11; + __m128i orig12; + __m128i orig13; + __m128i orig14 = x_14; + __m128i orig15 = x_15; + __m128i t_0, t_1, t_2, t_3, t_4, t_5, t_6, t_7, t_8, t_9, t_10, t_11, t_12, + t_13, t_14, t_15; + + uint32_t in12, in13; + int i; + + while (bytes >= 256) { + const __m128i addv12 = _mm_set_epi64x(1, 0); + const __m128i addv13 = _mm_set_epi64x(3, 2); + __m128i t12, t13; + uint64_t in1213; + + x_0 = orig0; + x_1 = orig1; + x_2 = orig2; + x_3 = orig3; + x_4 = orig4; + x_5 = orig5; + x_6 = orig6; + x_7 = orig7; + x_8 = orig8; + x_9 = orig9; + x_10 = orig10; + x_11 = orig11; + x_14 = orig14; + x_15 = orig15; + + in12 = x[12]; + in13 = x[13]; + in1213 = ((uint64_t) in12) | (((uint64_t) in13) << 32); + t12 = _mm_set1_epi64x(in1213); + t13 = _mm_set1_epi64x(in1213); + + x_12 = _mm_add_epi64(addv12, t12); + x_13 = _mm_add_epi64(addv13, t13); + + t12 = _mm_unpacklo_epi32(x_12, x_13); + t13 = _mm_unpackhi_epi32(x_12, x_13); + + x_12 = _mm_unpacklo_epi32(t12, t13); + x_13 = _mm_unpackhi_epi32(t12, t13); + + orig12 = x_12; + orig13 = x_13; + + in1213 += 4; + + x[12] = in1213 & 0xFFFFFFFF; + x[13] = (in1213 >> 32) & 0xFFFFFFFF; + + for (i = 0; i < ROUNDS; i += 2) { + VEC4_QUARTERROUND(0, 4, 8, 12); + VEC4_QUARTERROUND(1, 5, 9, 13); + VEC4_QUARTERROUND(2, 6, 10, 14); + VEC4_QUARTERROUND(3, 7, 11, 15); + VEC4_QUARTERROUND(0, 5, 10, 15); + VEC4_QUARTERROUND(1, 6, 11, 12); + VEC4_QUARTERROUND(2, 7, 8, 13); + VEC4_QUARTERROUND(3, 4, 9, 14); + } + +#define ONEQUAD_TRANSPOSE(A, B, C, D) \ + { \ + __m128i t0, t1, t2, t3; \ + \ + x_##A = _mm_add_epi32(x_##A, orig##A); \ + x_##B = _mm_add_epi32(x_##B, orig##B); \ + x_##C = _mm_add_epi32(x_##C, orig##C); \ + x_##D = _mm_add_epi32(x_##D, orig##D); \ + t_##A = _mm_unpacklo_epi32(x_##A, x_##B); \ + t_##B = _mm_unpacklo_epi32(x_##C, x_##D); \ + t_##C = _mm_unpackhi_epi32(x_##A, x_##B); \ + t_##D = _mm_unpackhi_epi32(x_##C, x_##D); \ + x_##A = _mm_unpacklo_epi64(t_##A, t_##B); \ + x_##B = _mm_unpackhi_epi64(t_##A, t_##B); \ + x_##C = _mm_unpacklo_epi64(t_##C, t_##D); \ + x_##D = _mm_unpackhi_epi64(t_##C, t_##D); \ + \ + t0 = _mm_xor_si128(x_##A, _mm_loadu_si128((const __m128i*) (m + 0))); \ + _mm_storeu_si128((__m128i*) (c + 0), t0); \ + t1 = _mm_xor_si128(x_##B, _mm_loadu_si128((const __m128i*) (m + 64))); \ + _mm_storeu_si128((__m128i*) (c + 64), t1); \ + t2 = \ + _mm_xor_si128(x_##C, _mm_loadu_si128((const __m128i*) (m + 128))); \ + _mm_storeu_si128((__m128i*) (c + 128), t2); \ + t3 = \ + _mm_xor_si128(x_##D, _mm_loadu_si128((const __m128i*) (m + 192))); \ + _mm_storeu_si128((__m128i*) (c + 192), t3); \ + } + +#define ONEQUAD(A, B, C, D) ONEQUAD_TRANSPOSE(A, B, C, D) + + ONEQUAD(0, 1, 2, 3); + m += 16; + c += 16; + ONEQUAD(4, 5, 6, 7); + m += 16; + c += 16; + ONEQUAD(8, 9, 10, 11); + m += 16; + c += 16; + ONEQUAD(12, 13, 14, 15); + m -= 48; + c -= 48; + +#undef ONEQUAD +#undef ONEQUAD_TRANSPOSE + + bytes -= 256; + c += 256; + m += 256; + } +} +#undef VEC4_ROT +#undef VEC4_QUARTERROUND +#undef VEC4_QUARTERROUND_SHUFFLE diff --git a/sodium/sodium_dolbeau_u8.h b/sodium/sodium_dolbeau_u8.h new file mode 100644 index 00000000..2f6c5f1e --- /dev/null +++ b/sodium/sodium_dolbeau_u8.h @@ -0,0 +1,357 @@ + +#define VEC8_ROT(A, IMM) \ + _mm256_or_si256(_mm256_slli_epi32(A, IMM), _mm256_srli_epi32(A, (32 - IMM))) + +/* implements a vector quarter round by-the-book (naive!) */ +#define VEC8_QUARTERROUND_NAIVE(A, B, C, D) \ + x_##A = _mm256_add_epi32(x_##A, x_##B); \ + t_##A = _mm256_xor_si256(x_##D, x_##A); \ + x_##D = VEC8_ROT(t_##A, 16); \ + x_##C = _mm256_add_epi32(x_##C, x_##D); \ + t_##C = _mm256_xor_si256(x_##B, x_##C); \ + x_##B = VEC8_ROT(t_##C, 12); \ + x_##A = _mm256_add_epi32(x_##A, x_##B); \ + t_##A = _mm256_xor_si256(x_##D, x_##A); \ + x_##D = VEC8_ROT(t_##A, 8); \ + x_##C = _mm256_add_epi32(x_##C, x_##D); \ + t_##C = _mm256_xor_si256(x_##B, x_##C); \ + x_##B = VEC8_ROT(t_##C, 7) + +/* same, but replace 2 of the shift/shift/or "rotation" by byte shuffles (8 & + * 16) (better) */ +#define VEC8_QUARTERROUND_SHUFFLE(A, B, C, D) \ + x_##A = _mm256_add_epi32(x_##A, x_##B); \ + t_##A = _mm256_xor_si256(x_##D, x_##A); \ + x_##D = _mm256_shuffle_epi8(t_##A, rot16); \ + x_##C = _mm256_add_epi32(x_##C, x_##D); \ + t_##C = _mm256_xor_si256(x_##B, x_##C); \ + x_##B = VEC8_ROT(t_##C, 12); \ + x_##A = _mm256_add_epi32(x_##A, x_##B); \ + t_##A = _mm256_xor_si256(x_##D, x_##A); \ + x_##D = _mm256_shuffle_epi8(t_##A, rot8); \ + x_##C = _mm256_add_epi32(x_##C, x_##D); \ + t_##C = _mm256_xor_si256(x_##B, x_##C); \ + x_##B = VEC8_ROT(t_##C, 7) + +/* same, but replace 2 of the shift/shift/or "rotation" by byte & word shuffles + * (8 & 16) (not as good as previous) */ +#define VEC8_QUARTERROUND_SHUFFLE2(A, B, C, D) \ + x_##A = _mm256_add_epi32(x_##A, x_##B); \ + t_##A = _mm256_xor_si256(x_##D, x_##A); \ + x_##D = _mm256_shufflehi_epi16(_mm256_shufflelo_epi16(t_##A, 0xb1), 0xb1); \ + x_##C = _mm256_add_epi32(x_##C, x_##D); \ + t_##C = _mm256_xor_si256(x_##B, x_##C); \ + x_##B = VEC8_ROT(t_##C, 12); \ + x_##A = _mm256_add_epi32(x_##A, x_##B); \ + t_##A = _mm256_xor_si256(x_##D, x_##A); \ + x_##D = _mm256_shuffle_epi8(t_##A, rot8); \ + x_##C = _mm256_add_epi32(x_##C, x_##D); \ + t_##C = _mm256_xor_si256(x_##B, x_##C); \ + x_##B = VEC8_ROT(t_##C, 7) + +#define VEC8_QUARTERROUND(A, B, C, D) VEC8_QUARTERROUND_SHUFFLE(A, B, C, D) + +#define VEC8_LINE1(A, B, C, D) \ + x_##A = _mm256_add_epi32(x_##A, x_##B); \ + x_##D = _mm256_shuffle_epi8(_mm256_xor_si256(x_##D, x_##A), rot16) +#define VEC8_LINE2(A, B, C, D) \ + x_##C = _mm256_add_epi32(x_##C, x_##D); \ + x_##B = VEC8_ROT(_mm256_xor_si256(x_##B, x_##C), 12) +#define VEC8_LINE3(A, B, C, D) \ + x_##A = _mm256_add_epi32(x_##A, x_##B); \ + x_##D = _mm256_shuffle_epi8(_mm256_xor_si256(x_##D, x_##A), rot8) +#define VEC8_LINE4(A, B, C, D) \ + x_##C = _mm256_add_epi32(x_##C, x_##D); \ + x_##B = VEC8_ROT(_mm256_xor_si256(x_##B, x_##C), 7) + +#define VEC8_ROUND_SEQ(A1, B1, C1, D1, A2, B2, C2, D2, A3, B3, C3, D3, A4, B4, \ + C4, D4) \ + VEC8_LINE1(A1, B1, C1, D1); \ + VEC8_LINE1(A2, B2, C2, D2); \ + VEC8_LINE1(A3, B3, C3, D3); \ + VEC8_LINE1(A4, B4, C4, D4); \ + VEC8_LINE2(A1, B1, C1, D1); \ + VEC8_LINE2(A2, B2, C2, D2); \ + VEC8_LINE2(A3, B3, C3, D3); \ + VEC8_LINE2(A4, B4, C4, D4); \ + VEC8_LINE3(A1, B1, C1, D1); \ + VEC8_LINE3(A2, B2, C2, D2); \ + VEC8_LINE3(A3, B3, C3, D3); \ + VEC8_LINE3(A4, B4, C4, D4); \ + VEC8_LINE4(A1, B1, C1, D1); \ + VEC8_LINE4(A2, B2, C2, D2); \ + VEC8_LINE4(A3, B3, C3, D3); \ + VEC8_LINE4(A4, B4, C4, D4) + +#define VEC8_ROUND_HALF(A1, B1, C1, D1, A2, B2, C2, D2, A3, B3, C3, D3, A4, \ + B4, C4, D4) \ + VEC8_LINE1(A1, B1, C1, D1); \ + VEC8_LINE1(A2, B2, C2, D2); \ + VEC8_LINE2(A1, B1, C1, D1); \ + VEC8_LINE2(A2, B2, C2, D2); \ + VEC8_LINE3(A1, B1, C1, D1); \ + VEC8_LINE3(A2, B2, C2, D2); \ + VEC8_LINE4(A1, B1, C1, D1); \ + VEC8_LINE4(A2, B2, C2, D2); \ + VEC8_LINE1(A3, B3, C3, D3); \ + VEC8_LINE1(A4, B4, C4, D4); \ + VEC8_LINE2(A3, B3, C3, D3); \ + VEC8_LINE2(A4, B4, C4, D4); \ + VEC8_LINE3(A3, B3, C3, D3); \ + VEC8_LINE3(A4, B4, C4, D4); \ + VEC8_LINE4(A3, B3, C3, D3); \ + VEC8_LINE4(A4, B4, C4, D4) + +#define VEC8_ROUND_HALFANDHALF(A1, B1, C1, D1, A2, B2, C2, D2, A3, B3, C3, D3, \ + A4, B4, C4, D4) \ + VEC8_LINE1(A1, B1, C1, D1); \ + VEC8_LINE1(A2, B2, C2, D2); \ + VEC8_LINE2(A1, B1, C1, D1); \ + VEC8_LINE2(A2, B2, C2, D2); \ + VEC8_LINE1(A3, B3, C3, D3); \ + VEC8_LINE1(A4, B4, C4, D4); \ + VEC8_LINE2(A3, B3, C3, D3); \ + VEC8_LINE2(A4, B4, C4, D4); \ + VEC8_LINE3(A1, B1, C1, D1); \ + VEC8_LINE3(A2, B2, C2, D2); \ + VEC8_LINE4(A1, B1, C1, D1); \ + VEC8_LINE4(A2, B2, C2, D2); \ + VEC8_LINE3(A3, B3, C3, D3); \ + VEC8_LINE3(A4, B4, C4, D4); \ + VEC8_LINE4(A3, B3, C3, D3); \ + VEC8_LINE4(A4, B4, C4, D4) + +#define VEC8_ROUND(A1, B1, C1, D1, A2, B2, C2, D2, A3, B3, C3, D3, A4, B4, C4, \ + D4) \ + VEC8_ROUND_SEQ(A1, B1, C1, D1, A2, B2, C2, D2, A3, B3, C3, D3, A4, B4, C4, \ + D4) + +if (bytes >= 512) { + /* constant for shuffling bytes (replacing multiple-of-8 rotates) */ + __m256i rot16 = + _mm256_set_epi8(13, 12, 15, 14, 9, 8, 11, 10, 5, 4, 7, 6, 1, 0, 3, 2, + 13, 12, 15, 14, 9, 8, 11, 10, 5, 4, 7, 6, 1, 0, 3, 2); + __m256i rot8 = + _mm256_set_epi8(14, 13, 12, 15, 10, 9, 8, 11, 6, 5, 4, 7, 2, 1, 0, 3, + 14, 13, 12, 15, 10, 9, 8, 11, 6, 5, 4, 7, 2, 1, 0, 3); + uint32_t in12, in13; + + /* the naive way seems as fast (if not a bit faster) than the vector way */ + __m256i x_0 = _mm256_set1_epi32(x[0]); + __m256i x_1 = _mm256_set1_epi32(x[1]); + __m256i x_2 = _mm256_set1_epi32(x[2]); + __m256i x_3 = _mm256_set1_epi32(x[3]); + __m256i x_4 = _mm256_set1_epi32(x[4]); + __m256i x_5 = _mm256_set1_epi32(x[5]); + __m256i x_6 = _mm256_set1_epi32(x[6]); + __m256i x_7 = _mm256_set1_epi32(x[7]); + __m256i x_8 = _mm256_set1_epi32(x[8]); + __m256i x_9 = _mm256_set1_epi32(x[9]); + __m256i x_10 = _mm256_set1_epi32(x[10]); + __m256i x_11 = _mm256_set1_epi32(x[11]); + __m256i x_12; + __m256i x_13; + __m256i x_14 = _mm256_set1_epi32(x[14]); + __m256i x_15 = _mm256_set1_epi32(x[15]); + + __m256i orig0 = x_0; + __m256i orig1 = x_1; + __m256i orig2 = x_2; + __m256i orig3 = x_3; + __m256i orig4 = x_4; + __m256i orig5 = x_5; + __m256i orig6 = x_6; + __m256i orig7 = x_7; + __m256i orig8 = x_8; + __m256i orig9 = x_9; + __m256i orig10 = x_10; + __m256i orig11 = x_11; + __m256i orig12; + __m256i orig13; + __m256i orig14 = x_14; + __m256i orig15 = x_15; + __m256i t_0, t_1, t_2, t_3, t_4, t_5, t_6, t_7, t_8, t_9, t_10, t_11, t_12, + t_13, t_14, t_15; + + while (bytes >= 512) { + const __m256i addv12 = _mm256_set_epi64x(3, 2, 1, 0); + const __m256i addv13 = _mm256_set_epi64x(7, 6, 5, 4); + const __m256i permute = _mm256_set_epi32(7, 6, 3, 2, 5, 4, 1, 0); + __m256i t12, t13; + + uint64_t in1213; + int i; + + x_0 = orig0; + x_1 = orig1; + x_2 = orig2; + x_3 = orig3; + x_4 = orig4; + x_5 = orig5; + x_6 = orig6; + x_7 = orig7; + x_8 = orig8; + x_9 = orig9; + x_10 = orig10; + x_11 = orig11; + x_14 = orig14; + x_15 = orig15; + + in12 = x[12]; + in13 = x[13]; + in1213 = ((uint64_t) in12) | (((uint64_t) in13) << 32); + x_12 = x_13 = _mm256_broadcastq_epi64(_mm_cvtsi64_si128(in1213)); + + t12 = _mm256_add_epi64(addv12, x_12); + t13 = _mm256_add_epi64(addv13, x_13); + + x_12 = _mm256_unpacklo_epi32(t12, t13); + x_13 = _mm256_unpackhi_epi32(t12, t13); + + t12 = _mm256_unpacklo_epi32(x_12, x_13); + t13 = _mm256_unpackhi_epi32(x_12, x_13); + + /* required because unpack* are intra-lane */ + x_12 = _mm256_permutevar8x32_epi32(t12, permute); + x_13 = _mm256_permutevar8x32_epi32(t13, permute); + + orig12 = x_12; + orig13 = x_13; + + in1213 += 8; + + x[12] = in1213 & 0xFFFFFFFF; + x[13] = (in1213 >> 32) & 0xFFFFFFFF; + + for (i = 0; i < ROUNDS; i += 2) { + VEC8_ROUND(0, 4, 8, 12, 1, 5, 9, 13, 2, 6, 10, 14, 3, 7, 11, 15); + VEC8_ROUND(0, 5, 10, 15, 1, 6, 11, 12, 2, 7, 8, 13, 3, 4, 9, 14); + } + +#define ONEQUAD_TRANSPOSE(A, B, C, D) \ + { \ + __m128i t0, t1, t2, t3; \ + x_##A = _mm256_add_epi32(x_##A, orig##A); \ + x_##B = _mm256_add_epi32(x_##B, orig##B); \ + x_##C = _mm256_add_epi32(x_##C, orig##C); \ + x_##D = _mm256_add_epi32(x_##D, orig##D); \ + t_##A = _mm256_unpacklo_epi32(x_##A, x_##B); \ + t_##B = _mm256_unpacklo_epi32(x_##C, x_##D); \ + t_##C = _mm256_unpackhi_epi32(x_##A, x_##B); \ + t_##D = _mm256_unpackhi_epi32(x_##C, x_##D); \ + x_##A = _mm256_unpacklo_epi64(t_##A, t_##B); \ + x_##B = _mm256_unpackhi_epi64(t_##A, t_##B); \ + x_##C = _mm256_unpacklo_epi64(t_##C, t_##D); \ + x_##D = _mm256_unpackhi_epi64(t_##C, t_##D); \ + t0 = _mm_xor_si128(_mm256_extracti128_si256(x_##A, 0), \ + _mm_loadu_si128((const __m128i*) (m + 0))); \ + _mm_storeu_si128((__m128i*) (c + 0), t0); \ + t1 = _mm_xor_si128(_mm256_extracti128_si256(x_##B, 0), \ + _mm_loadu_si128((const __m128i*) (m + 64))); \ + _mm_storeu_si128((__m128i*) (c + 64), t1); \ + t2 = _mm_xor_si128(_mm256_extracti128_si256(x_##C, 0), \ + _mm_loadu_si128((const __m128i*) (m + 128))); \ + _mm_storeu_si128((__m128i*) (c + 128), t2); \ + t3 = _mm_xor_si128(_mm256_extracti128_si256(x_##D, 0), \ + _mm_loadu_si128((const __m128i*) (m + 192))); \ + _mm_storeu_si128((__m128i*) (c + 192), t3); \ + t0 = _mm_xor_si128(_mm256_extracti128_si256(x_##A, 1), \ + _mm_loadu_si128((const __m128i*) (m + 256))); \ + _mm_storeu_si128((__m128i*) (c + 256), t0); \ + t1 = _mm_xor_si128(_mm256_extracti128_si256(x_##B, 1), \ + _mm_loadu_si128((const __m128i*) (m + 320))); \ + _mm_storeu_si128((__m128i*) (c + 320), t1); \ + t2 = _mm_xor_si128(_mm256_extracti128_si256(x_##C, 1), \ + _mm_loadu_si128((const __m128i*) (m + 384))); \ + _mm_storeu_si128((__m128i*) (c + 384), t2); \ + t3 = _mm_xor_si128(_mm256_extracti128_si256(x_##D, 1), \ + _mm_loadu_si128((const __m128i*) (m + 448))); \ + _mm_storeu_si128((__m128i*) (c + 448), t3); \ + } + +#define ONEQUAD(A, B, C, D) ONEQUAD_TRANSPOSE(A, B, C, D) + +#define ONEQUAD_UNPCK(A, B, C, D) \ + { \ + x_##A = _mm256_add_epi32(x_##A, orig##A); \ + x_##B = _mm256_add_epi32(x_##B, orig##B); \ + x_##C = _mm256_add_epi32(x_##C, orig##C); \ + x_##D = _mm256_add_epi32(x_##D, orig##D); \ + t_##A = _mm256_unpacklo_epi32(x_##A, x_##B); \ + t_##B = _mm256_unpacklo_epi32(x_##C, x_##D); \ + t_##C = _mm256_unpackhi_epi32(x_##A, x_##B); \ + t_##D = _mm256_unpackhi_epi32(x_##C, x_##D); \ + x_##A = _mm256_unpacklo_epi64(t_##A, t_##B); \ + x_##B = _mm256_unpackhi_epi64(t_##A, t_##B); \ + x_##C = _mm256_unpacklo_epi64(t_##C, t_##D); \ + x_##D = _mm256_unpackhi_epi64(t_##C, t_##D); \ + } + +#define ONEOCTO(A, B, C, D, A2, B2, C2, D2) \ + { \ + ONEQUAD_UNPCK(A, B, C, D); \ + ONEQUAD_UNPCK(A2, B2, C2, D2); \ + t_##A = _mm256_permute2x128_si256(x_##A, x_##A2, 0x20); \ + t_##A2 = _mm256_permute2x128_si256(x_##A, x_##A2, 0x31); \ + t_##B = _mm256_permute2x128_si256(x_##B, x_##B2, 0x20); \ + t_##B2 = _mm256_permute2x128_si256(x_##B, x_##B2, 0x31); \ + t_##C = _mm256_permute2x128_si256(x_##C, x_##C2, 0x20); \ + t_##C2 = _mm256_permute2x128_si256(x_##C, x_##C2, 0x31); \ + t_##D = _mm256_permute2x128_si256(x_##D, x_##D2, 0x20); \ + t_##D2 = _mm256_permute2x128_si256(x_##D, x_##D2, 0x31); \ + t_##A = _mm256_xor_si256( \ + t_##A, _mm256_loadu_si256((const __m256i*) (m + 0))); \ + t_##B = _mm256_xor_si256( \ + t_##B, _mm256_loadu_si256((const __m256i*) (m + 64))); \ + t_##C = _mm256_xor_si256( \ + t_##C, _mm256_loadu_si256((const __m256i*) (m + 128))); \ + t_##D = _mm256_xor_si256( \ + t_##D, _mm256_loadu_si256((const __m256i*) (m + 192))); \ + t_##A2 = _mm256_xor_si256( \ + t_##A2, _mm256_loadu_si256((const __m256i*) (m + 256))); \ + t_##B2 = _mm256_xor_si256( \ + t_##B2, _mm256_loadu_si256((const __m256i*) (m + 320))); \ + t_##C2 = _mm256_xor_si256( \ + t_##C2, _mm256_loadu_si256((const __m256i*) (m + 384))); \ + t_##D2 = _mm256_xor_si256( \ + t_##D2, _mm256_loadu_si256((const __m256i*) (m + 448))); \ + _mm256_storeu_si256((__m256i*) (c + 0), t_##A); \ + _mm256_storeu_si256((__m256i*) (c + 64), t_##B); \ + _mm256_storeu_si256((__m256i*) (c + 128), t_##C); \ + _mm256_storeu_si256((__m256i*) (c + 192), t_##D); \ + _mm256_storeu_si256((__m256i*) (c + 256), t_##A2); \ + _mm256_storeu_si256((__m256i*) (c + 320), t_##B2); \ + _mm256_storeu_si256((__m256i*) (c + 384), t_##C2); \ + _mm256_storeu_si256((__m256i*) (c + 448), t_##D2); \ + } + + ONEOCTO(0, 1, 2, 3, 4, 5, 6, 7); + m += 32; + c += 32; + ONEOCTO(8, 9, 10, 11, 12, 13, 14, 15); + m -= 32; + c -= 32; + +#undef ONEQUAD +#undef ONEQUAD_TRANSPOSE +#undef ONEQUAD_UNPCK +#undef ONEOCTO + + bytes -= 512; + c += 512; + m += 512; + } +} +#undef VEC8_ROT +#undef VEC8_QUARTERROUND +#undef VEC8_QUARTERROUND_NAIVE +#undef VEC8_QUARTERROUND_SHUFFLE +#undef VEC8_QUARTERROUND_SHUFFLE2 +#undef VEC8_LINE1 +#undef VEC8_LINE2 +#undef VEC8_LINE3 +#undef VEC8_LINE4 +#undef VEC8_ROUND +#undef VEC8_ROUND_SEQ +#undef VEC8_ROUND_HALF +#undef VEC8_ROUND_HALFANDHALF diff --git a/sodium/sodium_export.h b/sodium/sodium_export.h new file mode 100644 index 00000000..9d123849 --- /dev/null +++ b/sodium/sodium_export.h @@ -0,0 +1,27 @@ + +#ifndef sodium_export_H +#define sodium_export_H + +#include +#include +#include + +#if !defined(__clang__) && !defined(__GNUC__) +# ifdef __attribute__ +# undef __attribute__ +# endif +# define __attribute__(a) +#endif + +#ifndef CRYPTO_ALIGN +# if defined(__INTEL_COMPILER) || defined(_MSC_VER) +# define CRYPTO_ALIGN(x) __declspec(align(x)) +# else +# define CRYPTO_ALIGN(x) __attribute__ ((aligned(x))) +# endif +#endif + +#define SODIUM_MIN(A, B) ((A) < (B) ? (A) : (B)) +#define SODIUM_SIZE_MAX SODIUM_MIN(UINT64_MAX, SIZE_MAX) + +#endif diff --git a/sodium/sodium_fe_25_5_base.h b/sodium/sodium_fe_25_5_base.h new file mode 100644 index 00000000..e18530bb --- /dev/null +++ b/sodium/sodium_fe_25_5_base.h @@ -0,0 +1,1344 @@ +{ /* 0/31 */ + { + { 25967493, -14356035, 29566456, 3660896, -12694345, 4014787, 27544626, -11754271, -6079156, 2047605 }, + { -12545711, 934262, -2722910, 3049990, -727428, 9406986, 12720692, 5043384, 19500929, -15469378 }, + { -8738181, 4489570, 9688441, -14785194, 10184609, -12363380, 29287919, 11864899, -24514362, -4438546 } + }, + { + { -12815894, -12976347, -21581243, 11784320, -25355658, -2750717, -11717903, -3814571, -358445, -10211303 }, + { -21703237, 6903825, 27185491, 6451973, -29577724, -9554005, -15616551, 11189268, -26829678, -5319081 }, + { 26966642, 11152617, 32442495, 15396054, 14353839, -12752335, -3128826, -9541118, -15472047, -4166697 } + }, + { + { 15636291, -9688557, 24204773, -7912398, 616977, -16685262, 27787600, -14772189, 28944400, -1550024 }, + { 16568933, 4717097, -11556148, -1102322, 15682896, -11807043, 16354577, -11775962, 7689662, 11199574 }, + { 30464156, -5976125, -11779434, -15670865, 23220365, 15915852, 7512774, 10017326, -17749093, -9920357 } + }, + { + { -17036878, 13921892, 10945806, -6033431, 27105052, -16084379, -28926210, 15006023, 3284568, -6276540 }, + { 23599295, -8306047, -11193664, -7687416, 13236774, 10506355, 7464579, 9656445, 13059162, 10374397 }, + { 7798556, 16710257, 3033922, 2874086, 28997861, 2835604, 32406664, -3839045, -641708, -101325 } + }, + { + { 10861363, 11473154, 27284546, 1981175, -30064349, 12577861, 32867885, 14515107, -15438304, 10819380 }, + { 4708026, 6336745, 20377586, 9066809, -11272109, 6594696, -25653668, 12483688, -12668491, 5581306 }, + { 19563160, 16186464, -29386857, 4097519, 10237984, -4348115, 28542350, 13850243, -23678021, -15815942 } + }, + { + { -15371964, -12862754, 32573250, 4720197, -26436522, 5875511, -19188627, -15224819, -9818940, -12085777 }, + { -8549212, 109983, 15149363, 2178705, 22900618, 4543417, 3044240, -15689887, 1762328, 14866737 }, + { -18199695, -15951423, -10473290, 1707278, -17185920, 3916101, -28236412, 3959421, 27914454, 4383652 } + }, + { + { 5153746, 9909285, 1723747, -2777874, 30523605, 5516873, 19480852, 5230134, -23952439, -15175766 }, + { -30269007, -3463509, 7665486, 10083793, 28475525, 1649722, 20654025, 16520125, 30598449, 7715701 }, + { 28881845, 14381568, 9657904, 3680757, -20181635, 7843316, -31400660, 1370708, 29794553, -1409300 } + }, + { + { 14499471, -2729599, -33191113, -4254652, 28494862, 14271267, 30290735, 10876454, -33154098, 2381726 }, + { -7195431, -2655363, -14730155, 462251, -27724326, 3941372, -6236617, 3696005, -32300832, 15351955 }, + { 27431194, 8222322, 16448760, -3907995, -18707002, 11938355, -32961401, -2970515, 29551813, 10109425 } + } +}, +{ /* 1/31 */ + { + { -13657040, -13155431, -31283750, 11777098, 21447386, 6519384, -2378284, -1627556, 10092783, -4764171 }, + { 27939166, 14210322, 4677035, 16277044, -22964462, -12398139, -32508754, 12005538, -17810127, 12803510 }, + { 17228999, -15661624, -1233527, 300140, -1224870, -11714777, 30364213, -9038194, 18016357, 4397660 } + }, + { + { -10958843, -7690207, 4776341, -14954238, 27850028, -15602212, -26619106, 14544525, -17477504, 982639 }, + { 29253598, 15796703, -2863982, -9908884, 10057023, 3163536, 7332899, -4120128, -21047696, 9934963 }, + { 5793303, 16271923, -24131614, -10116404, 29188560, 1206517, -14747930, 4559895, -30123922, -10897950 } + }, + { + { -27643952, -11493006, 16282657, -11036493, 28414021, -15012264, 24191034, 4541697, -13338309, 5500568 }, + { 12650548, -1497113, 9052871, 11355358, -17680037, -8400164, -17430592, 12264343, 10874051, 13524335 }, + { 25556948, -3045990, 714651, 2510400, 23394682, -10415330, 33119038, 5080568, -22528059, 5376628 } + }, + { + { -26088264, -4011052, -17013699, -3537628, -6726793, 1920897, -22321305, -9447443, 4535768, 1569007 }, + { -2255422, 14606630, -21692440, -8039818, 28430649, 8775819, -30494562, 3044290, 31848280, 12543772 }, + { -22028579, 2943893, -31857513, 6777306, 13784462, -4292203, -27377195, -2062731, 7718482, 14474653 } + }, + { + { 2385315, 2454213, -22631320, 46603, -4437935, -15680415, 656965, -7236665, 24316168, -5253567 }, + { 13741529, 10911568, -33233417, -8603737, -20177830, -1033297, 33040651, -13424532, -20729456, 8321686 }, + { 21060490, -2212744, 15712757, -4336099, 1639040, 10656336, 23845965, -11874838, -9984458, 608372 } + }, + { + { -13672732, -15087586, -10889693, -7557059, -6036909, 11305547, 1123968, -6780577, 27229399, 23887 }, + { -23244140, -294205, -11744728, 14712571, -29465699, -2029617, 12797024, -6440308, -1633405, 16678954 }, + { -29500620, 4770662, -16054387, 14001338, 7830047, 9564805, -1508144, -4795045, -17169265, 4904953 } + }, + { + { 24059557, 14617003, 19037157, -15039908, 19766093, -14906429, 5169211, 16191880, 2128236, -4326833 }, + { -16981152, 4124966, -8540610, -10653797, 30336522, -14105247, -29806336, 916033, -6882542, -2986532 }, + { -22630907, 12419372, -7134229, -7473371, -16478904, 16739175, 285431, 2763829, 15736322, 4143876 } + }, + { + { 2379352, 11839345, -4110402, -5988665, 11274298, 794957, 212801, -14594663, 23527084, -16458268 }, + { 33431127, -11130478, -17838966, -15626900, 8909499, 8376530, -32625340, 4087881, -15188911, -14416214 }, + { 1767683, 7197987, -13205226, -2022635, -13091350, 448826, 5799055, 4357868, -4774191, -16323038 } + } +}, +{ /* 2/31 */ + { + { 6721966, 13833823, -23523388, -1551314, 26354293, -11863321, 23365147, -3949732, 7390890, 2759800 }, + { 4409041, 2052381, 23373853, 10530217, 7676779, -12885954, 21302353, -4264057, 1244380, -12919645 }, + { -4421239, 7169619, 4982368, -2957590, 30256825, -2777540, 14086413, 9208236, 15886429, 16489664 } + }, + { + { 1996075, 10375649, 14346367, 13311202, -6874135, -16438411, -13693198, 398369, -30606455, -712933 }, + { -25307465, 9795880, -2777414, 14878809, -33531835, 14780363, 13348553, 12076947, -30836462, 5113182 }, + { -17770784, 11797796, 31950843, 13929123, -25888302, 12288344, -30341101, -7336386, 13847711, 5387222 } + }, + { + { -18582163, -3416217, 17824843, -2340966, 22744343, -10442611, 8763061, 3617786, -19600662, 10370991 }, + { 20246567, -14369378, 22358229, -543712, 18507283, -10413996, 14554437, -8746092, 32232924, 16763880 }, + { 9648505, 10094563, 26416693, 14745928, -30374318, -6472621, 11094161, 15689506, 3140038, -16510092 } + }, + { + { -16160072, 5472695, 31895588, 4744994, 8823515, 10365685, -27224800, 9448613, -28774454, 366295 }, + { 19153450, 11523972, -11096490, -6503142, -24647631, 5420647, 28344573, 8041113, 719605, 11671788 }, + { 8678025, 2694440, -6808014, 2517372, 4964326, 11152271, -15432916, -15266516, 27000813, -10195553 } + }, + { + { -15157904, 7134312, 8639287, -2814877, -7235688, 10421742, 564065, 5336097, 6750977, -14521026 }, + { 11836410, -3979488, 26297894, 16080799, 23455045, 15735944, 1695823, -8819122, 8169720, 16220347 }, + { -18115838, 8653647, 17578566, -6092619, -8025777, -16012763, -11144307, -2627664, -5990708, -14166033 } + }, + { + { -23308498, -10968312, 15213228, -10081214, -30853605, -11050004, 27884329, 2847284, 2655861, 1738395 }, + { -27537433, -14253021, -25336301, -8002780, -9370762, 8129821, 21651608, -3239336, -19087449, -11005278 }, + { 1533110, 3437855, 23735889, 459276, 29970501, 11335377, 26030092, 5821408, 10478196, 8544890 } + }, + { + { 32173121, -16129311, 24896207, 3921497, 22579056, -3410854, 19270449, 12217473, 17789017, -3395995 }, + { -30552961, -2228401, -15578829, -10147201, 13243889, 517024, 15479401, -3853233, 30460520, 1052596 }, + { -11614875, 13323618, 32618793, 8175907, -15230173, 12596687, 27491595, -4612359, 3179268, -9478891 } + }, + { + { 31947069, -14366651, -4640583, -15339921, -15125977, -6039709, -14756777, -16411740, 19072640, -9511060 }, + { 11685058, 11822410, 3158003, -13952594, 33402194, -4165066, 5977896, -5215017, 473099, 5040608 }, + { -20290863, 8198642, -27410132, 11602123, 1290375, -2799760, 28326862, 1721092, -19558642, -3131606 } + } +}, +{ /* 3/31 */ + { + { 7881532, 10687937, 7578723, 7738378, -18951012, -2553952, 21820786, 8076149, -27868496, 11538389 }, + { -19935666, 3899861, 18283497, -6801568, -15728660, -11249211, 8754525, 7446702, -5676054, 5797016 }, + { -11295600, -3793569, -15782110, -7964573, 12708869, -8456199, 2014099, -9050574, -2369172, -5877341 } + }, + { + { -22472376, -11568741, -27682020, 1146375, 18956691, 16640559, 1192730, -3714199, 15123619, 10811505 }, + { 14352098, -3419715, -18942044, 10822655, 32750596, 4699007, -70363, 15776356, -28886779, -11974553 }, + { -28241164, -8072475, -4978962, -5315317, 29416931, 1847569, -20654173, -16484855, 4714547, -9600655 } + }, + { + { 15200332, 8368572, 19679101, 15970074, -31872674, 1959451, 24611599, -4543832, -11745876, 12340220 }, + { 12876937, -10480056, 33134381, 6590940, -6307776, 14872440, 9613953, 8241152, 15370987, 9608631 }, + { -4143277, -12014408, 8446281, -391603, 4407738, 13629032, -7724868, 15866074, -28210621, -8814099 } + }, + { + { 26660628, -15677655, 8393734, 358047, -7401291, 992988, -23904233, 858697, 20571223, 8420556 }, + { 14620715, 13067227, -15447274, 8264467, 14106269, 15080814, 33531827, 12516406, -21574435, -12476749 }, + { 236881, 10476226, 57258, -14677024, 6472998, 2466984, 17258519, 7256740, 8791136, 15069930 } + }, + { + { 1276410, -9371918, 22949635, -16322807, -23493039, -5702186, 14711875, 4874229, -30663140, -2331391 }, + { 5855666, 4990204, -13711848, 7294284, -7804282, 1924647, -1423175, -7912378, -33069337, 9234253 }, + { 20590503, -9018988, 31529744, -7352666, -2706834, 10650548, 31559055, -11609587, 18979186, 13396066 } + }, + { + { 24474287, 4968103, 22267082, 4407354, 24063882, -8325180, -18816887, 13594782, 33514650, 7021958 }, + { -11566906, -6565505, -21365085, 15928892, -26158305, 4315421, -25948728, -3916677, -21480480, 12868082 }, + { -28635013, 13504661, 19988037, -2132761, 21078225, 6443208, -21446107, 2244500, -12455797, -8089383 } + }, + { + { -30595528, 13793479, -5852820, 319136, -25723172, -6263899, 33086546, 8957937, -15233648, 5540521 }, + { -11630176, -11503902, -8119500, -7643073, 2620056, 1022908, -23710744, -1568984, -16128528, -14962807 }, + { 23152971, 775386, 27395463, 14006635, -9701118, 4649512, 1689819, 892185, -11513277, -15205948 } + }, + { + { 9770129, 9586738, 26496094, 4324120, 1556511, -3550024, 27453819, 4763127, -19179614, 5867134 }, + { -32765025, 1927590, 31726409, -4753295, 23962434, -16019500, 27846559, 5931263, -29749703, -16108455 }, + { 27461885, -2977536, 22380810, 1815854, -23033753, -3031938, 7283490, -15148073, -19526700, 7734629 } + } +}, +{ /* 4/31 */ + { + { -8010264, -9590817, -11120403, 6196038, 29344158, -13430885, 7585295, -3176626, 18549497, 15302069 }, + { -32658337, -6171222, -7672793, -11051681, 6258878, 13504381, 10458790, -6418461, -8872242, 8424746 }, + { 24687205, 8613276, -30667046, -3233545, 1863892, -1830544, 19206234, 7134917, -11284482, -828919 } + }, + { + { 11334899, -9218022, 8025293, 12707519, 17523892, -10476071, 10243738, -14685461, -5066034, 16498837 }, + { 8911542, 6887158, -9584260, -6958590, 11145641, -9543680, 17303925, -14124238, 6536641, 10543906 }, + { -28946384, 15479763, -17466835, 568876, -1497683, 11223454, -2669190, -16625574, -27235709, 8876771 } + }, + { + { -25742899, -12566864, -15649966, -846607, -33026686, -796288, -33481822, 15824474, -604426, -9039817 }, + { 10330056, 70051, 7957388, -9002667, 9764902, 15609756, 27698697, -4890037, 1657394, 3084098 }, + { 10477963, -7470260, 12119566, -13250805, 29016247, -5365589, 31280319, 14396151, -30233575, 15272409 } + }, + { + { -12288309, 3169463, 28813183, 16658753, 25116432, -5630466, -25173957, -12636138, -25014757, 1950504 }, + { -26180358, 9489187, 11053416, -14746161, -31053720, 5825630, -8384306, -8767532, 15341279, 8373727 }, + { 28685821, 7759505, -14378516, -12002860, -31971820, 4079242, 298136, -10232602, -2878207, 15190420 } + }, + { + { -32932876, 13806336, -14337485, -15794431, -24004620, 10940928, 8669718, 2742393, -26033313, -6875003 }, + { -1580388, -11729417, -25979658, -11445023, -17411874, -10912854, 9291594, -16247779, -12154742, 6048605 }, + { -30305315, 14843444, 1539301, 11864366, 20201677, 1900163, 13934231, 5128323, 11213262, 9168384 } + }, + { + { -26280513, 11007847, 19408960, -940758, -18592965, -4328580, -5088060, -11105150, 20470157, -16398701 }, + { -23136053, 9282192, 14855179, -15390078, -7362815, -14408560, -22783952, 14461608, 14042978, 5230683 }, + { 29969567, -2741594, -16711867, -8552442, 9175486, -2468974, 21556951, 3506042, -5933891, -12449708 } + }, + { + { -3144746, 8744661, 19704003, 4581278, -20430686, 6830683, -21284170, 8971513, -28539189, 15326563 }, + { -19464629, 10110288, -17262528, -3503892, -23500387, 1355669, -15523050, 15300988, -20514118, 9168260 }, + { -5353335, 4488613, -23803248, 16314347, 7780487, -15638939, -28948358, 9601605, 33087103, -9011387 } + }, + { + { -19443170, -15512900, -20797467, -12445323, -29824447, 10229461, -27444329, -15000531, -5996870, 15664672 }, + { 23294591, -16632613, -22650781, -8470978, 27844204, 11461195, 13099750, -2460356, 18151676, 13417686 }, + { -24722913, -4176517, -31150679, 5988919, -26858785, 6685065, 1661597, -12551441, 15271676, -15452665 } + } +}, +{ /* 5/31 */ + { + { 11433042, -13228665, 8239631, -5279517, -1985436, -725718, -18698764, 2167544, -6921301, -13440182 }, + { -31436171, 15575146, 30436815, 12192228, -22463353, 9395379, -9917708, -8638997, 12215110, 12028277 }, + { 14098400, 6555944, 23007258, 5757252, -15427832, -12950502, 30123440, 4617780, -16900089, -655628 } + }, + { + { -4026201, -15240835, 11893168, 13718664, -14809462, 1847385, -15819999, 10154009, 23973261, -12684474 }, + { -26531820, -3695990, -1908898, 2534301, -31870557, -16550355, 18341390, -11419951, 32013174, -10103539 }, + { -25479301, 10876443, -11771086, -14625140, -12369567, 1838104, 21911214, 6354752, 4425632, -837822 } + }, + { + { -10433389, -14612966, 22229858, -3091047, -13191166, 776729, -17415375, -12020462, 4725005, 14044970 }, + { 19268650, -7304421, 1555349, 8692754, -21474059, -9910664, 6347390, -1411784, -19522291, -16109756 }, + { -24864089, 12986008, -10898878, -5558584, -11312371, -148526, 19541418, 8180106, 9282262, 10282508 } + }, + { + { -26205082, 4428547, -8661196, -13194263, 4098402, -14165257, 15522535, 8372215, 5542595, -10702683 }, + { -10562541, 14895633, 26814552, -16673850, -17480754, -2489360, -2781891, 6993761, -18093885, 10114655 }, + { -20107055, -929418, 31422704, 10427861, -7110749, 6150669, -29091755, -11529146, 25953725, -106158 } + }, + { + { -4234397, -8039292, -9119125, 3046000, 2101609, -12607294, 19390020, 6094296, -3315279, 12831125 }, + { -15998678, 7578152, 5310217, 14408357, -33548620, -224739, 31575954, 6326196, 7381791, -2421839 }, + { -20902779, 3296811, 24736065, -16328389, 18374254, 7318640, 6295303, 8082724, -15362489, 12339664 } + }, + { + { 27724736, 2291157, 6088201, -14184798, 1792727, 5857634, 13848414, 15768922, 25091167, 14856294 }, + { -18866652, 8331043, 24373479, 8541013, -701998, -9269457, 12927300, -12695493, -22182473, -9012899 }, + { -11423429, -5421590, 11632845, 3405020, 30536730, -11674039, -27260765, 13866390, 30146206, 9142070 } + }, + { + { 3924129, -15307516, -13817122, -10054960, 12291820, -668366, -27702774, 9326384, -8237858, 4171294 }, + { -15921940, 16037937, 6713787, 16606682, -21612135, 2790944, 26396185, 3731949, 345228, -5462949 }, + { -21327538, 13448259, 25284571, 1143661, 20614966, -8849387, 2031539, -12391231, -16253183, -13582083 } + }, + { + { 31016211, -16722429, 26371392, -14451233, -5027349, 14854137, 17477601, 3842657, 28012650, -16405420 }, + { -5075835, 9368966, -8562079, -4600902, -15249953, 6970560, -9189873, 16292057, -8867157, 3507940 }, + { 29439664, 3537914, 23333589, 6997794, -17555561, -11018068, -15209202, -15051267, -9164929, 6580396 } + } +}, +{ /* 6/31 */ + { + { -12185861, -7679788, 16438269, 10826160, -8696817, -6235611, 17860444, -9273846, -2095802, 9304567 }, + { 20714564, -4336911, 29088195, 7406487, 11426967, -5095705, 14792667, -14608617, 5289421, -477127 }, + { -16665533, -10650790, -6160345, -13305760, 9192020, -1802462, 17271490, 12349094, 26939669, -3752294 } + }, + { + { -12889898, 9373458, 31595848, 16374215, 21471720, 13221525, -27283495, -12348559, -3698806, 117887 }, + { 22263325, -6560050, 3984570, -11174646, -15114008, -566785, 28311253, 5358056, -23319780, 541964 }, + { 16259219, 3261970, 2309254, -15534474, -16885711, -4581916, 24134070, -16705829, -13337066, -13552195 } + }, + { + { 9378160, -13140186, -22845982, -12745264, 28198281, -7244098, -2399684, -717351, 690426, 14876244 }, + { 24977353, -314384, -8223969, -13465086, 28432343, -1176353, -13068804, -12297348, -22380984, 6618999 }, + { -1538174, 11685646, 12944378, 13682314, -24389511, -14413193, 8044829, -13817328, 32239829, -5652762 } + }, + { + { -18603066, 4762990, -926250, 8885304, -28412480, -3187315, 9781647, -10350059, 32779359, 5095274 }, + { -33008130, -5214506, -32264887, -3685216, 9460461, -9327423, -24601656, 14506724, 21639561, -2630236 }, + { -16400943, -13112215, 25239338, 15531969, 3987758, -4499318, -1289502, -6863535, 17874574, 558605 } + }, + { + { -13600129, 10240081, 9171883, 16131053, -20869254, 9599700, 33499487, 5080151, 2085892, 5119761 }, + { -22205145, -2519528, -16381601, 414691, -25019550, 2170430, 30634760, -8363614, -31999993, -5759884 }, + { -6845704, 15791202, 8550074, -1312654, 29928809, -12092256, 27534430, -7192145, -22351378, 12961482 } + }, + { + { -24492060, -9570771, 10368194, 11582341, -23397293, -2245287, 16533930, 8206996, -30194652, -5159638 }, + { -11121496, -3382234, 2307366, 6362031, -135455, 8868177, -16835630, 7031275, 7589640, 8945490 }, + { -32152748, 8917967, 6661220, -11677616, -1192060, -15793393, 7251489, -11182180, 24099109, -14456170 } + }, + { + { 5019558, -7907470, 4244127, -14714356, -26933272, 6453165, -19118182, -13289025, -6231896, -10280736 }, + { 10853594, 10721687, 26480089, 5861829, -22995819, 1972175, -1866647, -10557898, -3363451, -6441124 }, + { -17002408, 5906790, 221599, -6563147, 7828208, -13248918, 24362661, -2008168, -13866408, 7421392 } + }, + { + { 8139927, -6546497, 32257646, -5890546, 30375719, 1886181, -21175108, 15441252, 28826358, -4123029 }, + { 6267086, 9695052, 7709135, -16603597, -32869068, -1886135, 14795160, -7840124, 13746021, -1742048 }, + { 28584902, 7787108, -6732942, -15050729, 22846041, -7571236, -3181936, -363524, 4771362, -8419958 } + } +}, +{ /* 7/31 */ + { + { 24949256, 6376279, -27466481, -8174608, -18646154, -9930606, 33543569, -12141695, 3569627, 11342593 }, + { 26514989, 4740088, 27912651, 3697550, 19331575, -11472339, 6809886, 4608608, 7325975, -14801071 }, + { -11618399, -14554430, -24321212, 7655128, -1369274, 5214312, -27400540, 10258390, -17646694, -8186692 } + }, + { + { 11431204, 15823007, 26570245, 14329124, 18029990, 4796082, -31446179, 15580664, 9280358, -3973687 }, + { -160783, -10326257, -22855316, -4304997, -20861367, -13621002, -32810901, -11181622, -15545091, 4387441 }, + { -20799378, 12194512, 3937617, -5805892, -27154820, 9340370, -24513992, 8548137, 20617071, -7482001 } + }, + { + { -938825, -3930586, -8714311, 16124718, 24603125, -6225393, -13775352, -11875822, 24345683, 10325460 }, + { -19855277, -1568885, -22202708, 8714034, 14007766, 6928528, 16318175, -1010689, 4766743, 3552007 }, + { -21751364, -16730916, 1351763, -803421, -4009670, 3950935, 3217514, 14481909, 10988822, -3994762 } + }, + { + { 15564307, -14311570, 3101243, 5684148, 30446780, -8051356, 12677127, -6505343, -8295852, 13296005 }, + { -9442290, 6624296, -30298964, -11913677, -4670981, -2057379, 31521204, 9614054, -30000824, 12074674 }, + { 4771191, -135239, 14290749, -13089852, 27992298, 14998318, -1413936, -1556716, 29832613, -16391035 } + }, + { + { 7064884, -7541174, -19161962, -5067537, -18891269, -2912736, 25825242, 5293297, -27122660, 13101590 }, + { -2298563, 2439670, -7466610, 1719965, -27267541, -16328445, 32512469, -5317593, -30356070, -4190957 }, + { -30006540, 10162316, -33180176, 3981723, -16482138, -13070044, 14413974, 9515896, 19568978, 9628812 } + }, + { + { 33053803, 199357, 15894591, 1583059, 27380243, -4580435, -17838894, -6106839, -6291786, 3437740 }, + { -18978877, 3884493, 19469877, 12726490, 15913552, 13614290, -22961733, 70104, 7463304, 4176122 }, + { -27124001, 10659917, 11482427, -16070381, 12771467, -6635117, -32719404, -5322751, 24216882, 5944158 } + }, + { + { 8894125, 7450974, -2664149, -9765752, -28080517, -12389115, 19345746, 14680796, 11632993, 5847885 }, + { 26942781, -2315317, 9129564, -4906607, 26024105, 11769399, -11518837, 6367194, -9727230, 4782140 }, + { 19916461, -4828410, -22910704, -11414391, 25606324, -5972441, 33253853, 8220911, 6358847, -1873857 } + }, + { + { 801428, -2081702, 16569428, 11065167, 29875704, 96627, 7908388, -4480480, -13538503, 1387155 }, + { 19646058, 5720633, -11416706, 12814209, 11607948, 12749789, 14147075, 15156355, -21866831, 11835260 }, + { 19299512, 1155910, 28703737, 14890794, 2925026, 7269399, 26121523, 15467869, -26560550, 5052483 } + } +}, +{ /* 8/31 */ + { + { -3017432, 10058206, 1980837, 3964243, 22160966, 12322533, -6431123, -12618185, 12228557, -7003677 }, + { 32944382, 14922211, -22844894, 5188528, 21913450, -8719943, 4001465, 13238564, -6114803, 8653815 }, + { 22865569, -4652735, 27603668, -12545395, 14348958, 8234005, 24808405, 5719875, 28483275, 2841751 } + }, + { + { -16420968, -1113305, -327719, -12107856, 21886282, -15552774, -1887966, -315658, 19932058, -12739203 }, + { -11656086, 10087521, -8864888, -5536143, -19278573, -3055912, 3999228, 13239134, -4777469, -13910208 }, + { 1382174, -11694719, 17266790, 9194690, -13324356, 9720081, 20403944, 11284705, -14013818, 3093230 } + }, + { + { 16650921, -11037932, -1064178, 1570629, -8329746, 7352753, -302424, 16271225, -24049421, -6691850 }, + { -21911077, -5927941, -4611316, -5560156, -31744103, -10785293, 24123614, 15193618, -21652117, -16739389 }, + { -9935934, -4289447, -25279823, 4372842, 2087473, 10399484, 31870908, 14690798, 17361620, 11864968 } + }, + { + { -11307610, 6210372, 13206574, 5806320, -29017692, -13967200, -12331205, -7486601, -25578460, -16240689 }, + { 14668462, -12270235, 26039039, 15305210, 25515617, 4542480, 10453892, 6577524, 9145645, -6443880 }, + { 5974874, 3053895, -9433049, -10385191, -31865124, 3225009, -7972642, 3936128, -5652273, -3050304 } + }, + { + { 30625386, -4729400, -25555961, -12792866, -20484575, 7695099, 17097188, -16303496, -27999779, 1803632 }, + { -3553091, 9865099, -5228566, 4272701, -5673832, -16689700, 14911344, 12196514, -21405489, 7047412 }, + { 20093277, 9920966, -11138194, -5343857, 13161587, 12044805, -32856851, 4124601, -32343828, -10257566 } + }, + { + { -20788824, 14084654, -13531713, 7842147, 19119038, -13822605, 4752377, -8714640, -21679658, 2288038 }, + { -26819236, -3283715, 29965059, 3039786, -14473765, 2540457, 29457502, 14625692, -24819617, 12570232 }, + { -1063558, -11551823, 16920318, 12494842, 1278292, -5869109, -21159943, -3498680, -11974704, 4724943 } + }, + { + { 17960970, -11775534, -4140968, -9702530, -8876562, -1410617, -12907383, -8659932, -29576300, 1903856 }, + { 23134274, -14279132, -10681997, -1611936, 20684485, 15770816, -12989750, 3190296, 26955097, 14109738 }, + { 15308788, 5320727, -30113809, -14318877, 22902008, 7767164, 29425325, -11277562, 31960942, 11934971 } + }, + { + { -27395711, 8435796, 4109644, 12222639, -24627868, 14818669, 20638173, 4875028, 10491392, 1379718 }, + { -13159415, 9197841, 3875503, -8936108, -1383712, -5879801, 33518459, 16176658, 21432314, 12180697 }, + { -11787308, 11500838, 13787581, -13832590, -22430679, 10140205, 1465425, 12689540, -10301319, -13872883 } + } +}, +{ /* 9/31 */ + { + { 5414091, -15386041, -21007664, 9643570, 12834970, 1186149, -2622916, -1342231, 26128231, 6032912 }, + { -26337395, -13766162, 32496025, -13653919, 17847801, -12669156, 3604025, 8316894, -25875034, -10437358 }, + { 3296484, 6223048, 24680646, -12246460, -23052020, 5903205, -8862297, -4639164, 12376617, 3188849 } + }, + { + { 29190488, -14659046, 27549113, -1183516, 3520066, -10697301, 32049515, -7309113, -16109234, -9852307 }, + { -14744486, -9309156, 735818, -598978, -20407687, -5057904, 25246078, -15795669, 18640741, -960977 }, + { -6928835, -16430795, 10361374, 5642961, 4910474, 12345252, -31638386, -494430, 10530747, 1053335 } + }, + { + { -29265967, -14186805, -13538216, -12117373, -19457059, -10655384, -31462369, -2948985, 24018831, 15026644 }, + { -22592535, -3145277, -2289276, 5953843, -13440189, 9425631, 25310643, 13003497, -2314791, -15145616 }, + { -27419985, -603321, -8043984, -1669117, -26092265, 13987819, -27297622, 187899, -23166419, -2531735 } + }, + { + { -21744398, -13810475, 1844840, 5021428, -10434399, -15911473, 9716667, 16266922, -5070217, 726099 }, + { 29370922, -6053998, 7334071, -15342259, 9385287, 2247707, -13661962, -4839461, 30007388, -15823341 }, + { -936379, 16086691, 23751945, -543318, -1167538, -5189036, 9137109, 730663, 9835848, 4555336 } + }, + { + { -23376435, 1410446, -22253753, -12899614, 30867635, 15826977, 17693930, 544696, -11985298, 12422646 }, + { 31117226, -12215734, -13502838, 6561947, -9876867, -12757670, -5118685, -4096706, 29120153, 13924425 }, + { -17400879, -14233209, 19675799, -2734756, -11006962, -5858820, -9383939, -11317700, 7240931, -237388 } + }, + { + { -31361739, -11346780, -15007447, -5856218, -22453340, -12152771, 1222336, 4389483, 3293637, -15551743 }, + { -16684801, -14444245, 11038544, 11054958, -13801175, -3338533, -24319580, 7733547, 12796905, -6335822 }, + { -8759414, -10817836, -25418864, 10783769, -30615557, -9746811, -28253339, 3647836, 3222231, -11160462 } + }, + { + { 18606113, 1693100, -25448386, -15170272, 4112353, 10045021, 23603893, -2048234, -7550776, 2484985 }, + { 9255317, -3131197, -12156162, -1004256, 13098013, -9214866, 16377220, -2102812, -19802075, -3034702 }, + { -22729289, 7496160, -5742199, 11329249, 19991973, -3347502, -31718148, 9936966, -30097688, -10618797 } + }, + { + { 21878590, -5001297, 4338336, 13643897, -3036865, 13160960, 19708896, 5415497, -7360503, -4109293 }, + { 27736861, 10103576, 12500508, 8502413, -3413016, -9633558, 10436918, -1550276, -23659143, -8132100 }, + { 19492550, -12104365, -29681976, -852630, -3208171, 12403437, 30066266, 8367329, 13243957, 8709688 } + } +}, +{ /* 10/31 */ + { + { 12015105, 2801261, 28198131, 10151021, 24818120, -4743133, -11194191, -5645734, 5150968, 7274186 }, + { 2831366, -12492146, 1478975, 6122054, 23825128, -12733586, 31097299, 6083058, 31021603, -9793610 }, + { -2529932, -2229646, 445613, 10720828, -13849527, -11505937, -23507731, 16354465, 15067285, -14147707 } + }, + { + { 7840942, 14037873, -33364863, 15934016, -728213, -3642706, 21403988, 1057586, -19379462, -12403220 }, + { 915865, -16469274, 15608285, -8789130, -24357026, 6060030, -17371319, 8410997, -7220461, 16527025 }, + { 32922597, -556987, 20336074, -16184568, 10903705, -5384487, 16957574, 52992, 23834301, 6588044 } + }, + { + { 32752030, 11232950, 3381995, -8714866, 22652988, -10744103, 17159699, 16689107, -20314580, -1305992 }, + { -4689649, 9166776, -25710296, -10847306, 11576752, 12733943, 7924251, -2752281, 1976123, -7249027 }, + { 21251222, 16309901, -2983015, -6783122, 30810597, 12967303, 156041, -3371252, 12331345, -8237197 } + }, + { + { 8651614, -4477032, -16085636, -4996994, 13002507, 2950805, 29054427, -5106970, 10008136, -4667901 }, + { 31486080, 15114593, -14261250, 12951354, 14369431, -7387845, 16347321, -13662089, 8684155, -10532952 }, + { 19443825, 11385320, 24468943, -9659068, -23919258, 2187569, -26263207, -6086921, 31316348, 14219878 } + }, + { + { -28594490, 1193785, 32245219, 11392485, 31092169, 15722801, 27146014, 6992409, 29126555, 9207390 }, + { 32382935, 1110093, 18477781, 11028262, -27411763, -7548111, -4980517, 10843782, -7957600, -14435730 }, + { 2814918, 7836403, 27519878, -7868156, -20894015, -11553689, -21494559, 8550130, 28346258, 1994730 } + }, + { + { -19578299, 8085545, -14000519, -3948622, 2785838, -16231307, -19516951, 7174894, 22628102, 8115180 }, + { -30405132, 955511, -11133838, -15078069, -32447087, -13278079, -25651578, 3317160, -9943017, 930272 }, + { -15303681, -6833769, 28856490, 1357446, 23421993, 1057177, 24091212, -1388970, -22765376, -10650715 } + }, + { + { -22751231, -5303997, -12907607, -12768866, -15811511, -7797053, -14839018, -16554220, -1867018, 8398970 }, + { -31969310, 2106403, -4736360, 1362501, 12813763, 16200670, 22981545, -6291273, 18009408, -15772772 }, + { -17220923, -9545221, -27784654, 14166835, 29815394, 7444469, 29551787, -3727419, 19288549, 1325865 } + }, + { + { 15100157, -15835752, -23923978, -1005098, -26450192, 15509408, 12376730, -3479146, 33166107, -8042750 }, + { 20909231, 13023121, -9209752, 16251778, -5778415, -8094914, 12412151, 10018715, 2213263, -13878373 }, + { 32529814, -11074689, 30361439, -16689753, -9135940, 1513226, 22922121, 6382134, -5766928, 8371348 } + } +}, +{ /* 11/31 */ + { + { 9923462, 11271500, 12616794, 3544722, -29998368, -1721626, 12891687, -8193132, -26442943, 10486144 }, + { -22597207, -7012665, 8587003, -8257861, 4084309, -12970062, 361726, 2610596, -23921530, -11455195 }, + { 5408411, -1136691, -4969122, 10561668, 24145918, 14240566, 31319731, -4235541, 19985175, -3436086 } + }, + { + { -13994457, 16616821, 14549246, 3341099, 32155958, 13648976, -17577068, 8849297, 65030, 8370684 }, + { -8320926, -12049626, 31204563, 5839400, -20627288, -1057277, -19442942, 6922164, 12743482, -9800518 }, + { -2361371, 12678785, 28815050, 4759974, -23893047, 4884717, 23783145, 11038569, 18800704, 255233 } + }, + { + { -5269658, -1773886, 13957886, 7990715, 23132995, 728773, 13393847, 9066957, 19258688, -14753793 }, + { -2936654, -10827535, -10432089, 14516793, -3640786, 4372541, -31934921, 2209390, -1524053, 2055794 }, + { 580882, 16705327, 5468415, -2683018, -30926419, -14696000, -7203346, -8994389, -30021019, 7394435 } + }, + { + { 23838809, 1822728, -15738443, 15242727, 8318092, -3733104, -21672180, -3492205, -4821741, 14799921 }, + { 13345610, 9759151, 3371034, -16137791, 16353039, 8577942, 31129804, 13496856, -9056018, 7402518 }, + { 2286874, -4435931, -20042458, -2008336, -13696227, 5038122, 11006906, -15760352, 8205061, 1607563 } + }, + { + { 14414086, -8002132, 3331830, -3208217, 22249151, -5594188, 18364661, -2906958, 30019587, -9029278 }, + { -27688051, 1585953, -10775053, 931069, -29120221, -11002319, -14410829, 12029093, 9944378, 8024 }, + { 4368715, -3709630, 29874200, -15022983, -20230386, -11410704, -16114594, -999085, -8142388, 5640030 } + }, + { + { 10299610, 13746483, 11661824, 16234854, 7630238, 5998374, 9809887, -16694564, 15219798, -14327783 }, + { 27425505, -5719081, 3055006, 10660664, 23458024, 595578, -15398605, -1173195, -18342183, 9742717 }, + { 6744077, 2427284, 26042789, 2720740, -847906, 1118974, 32324614, 7406442, 12420155, 1994844 } + }, + { + { 14012521, -5024720, -18384453, -9578469, -26485342, -3936439, -13033478, -10909803, 24319929, -6446333 }, + { 16412690, -4507367, 10772641, 15929391, -17068788, -4658621, 10555945, -10484049, -30102368, -4739048 }, + { 22397382, -7767684, -9293161, -12792868, 17166287, -9755136, -27333065, 6199366, 21880021, -12250760 } + }, + { + { -4283307, 5368523, -31117018, 8163389, -30323063, 3209128, 16557151, 8890729, 8840445, 4957760 }, + { -15447727, 709327, -6919446, -10870178, -29777922, 6522332, -21720181, 12130072, -14796503, 5005757 }, + { -2114751, -14308128, 23019042, 15765735, -25269683, 6002752, 10183197, -13239326, -16395286, -2176112 } + } +}, +{ /* 12/31 */ + { + { -19025756, 1632005, 13466291, -7995100, -23640451, 16573537, -32013908, -3057104, 22208662, 2000468 }, + { 3065073, -1412761, -25598674, -361432, -17683065, -5703415, -8164212, 11248527, -3691214, -7414184 }, + { 10379208, -6045554, 8877319, 1473647, -29291284, -12507580, 16690915, 2553332, -3132688, 16400289 } + }, + { + { 15716668, 1254266, -18472690, 7446274, -8448918, 6344164, -22097271, -7285580, 26894937, 9132066 }, + { 24158887, 12938817, 11085297, -8177598, -28063478, -4457083, -30576463, 64452, -6817084, -2692882 }, + { 13488534, 7794716, 22236231, 5989356, 25426474, -12578208, 2350710, -3418511, -4688006, 2364226 } + }, + { + { 16335052, 9132434, 25640582, 6678888, 1725628, 8517937, -11807024, -11697457, 15445875, -7798101 }, + { 29004207, -7867081, 28661402, -640412, -12794003, -7943086, 31863255, -4135540, -278050, -15759279 }, + { -6122061, -14866665, -28614905, 14569919, -10857999, -3591829, 10343412, -6976290, -29828287, -10815811 } + }, + { + { 27081650, 3463984, 14099042, -4517604, 1616303, -6205604, 29542636, 15372179, 17293797, 960709 }, + { 20263915, 11434237, -5765435, 11236810, 13505955, -10857102, -16111345, 6493122, -19384511, 7639714 }, + { -2830798, -14839232, 25403038, -8215196, -8317012, -16173699, 18006287, -16043750, 29994677, -15808121 } + }, + { + { 9769828, 5202651, -24157398, -13631392, -28051003, -11561624, -24613141, -13860782, -31184575, 709464 }, + { 12286395, 13076066, -21775189, -1176622, -25003198, 4057652, -32018128, -8890874, 16102007, 13205847 }, + { 13733362, 5599946, 10557076, 3195751, -5557991, 8536970, -25540170, 8525972, 10151379, 10394400 } + }, + { + { 4024660, -16137551, 22436262, 12276534, -9099015, -2686099, 19698229, 11743039, -33302334, 8934414 }, + { -15879800, -4525240, -8580747, -2934061, 14634845, -698278, -9449077, 3137094, -11536886, 11721158 }, + { 17555939, -5013938, 8268606, 2331751, -22738815, 9761013, 9319229, 8835153, -9205489, -1280045 } + }, + { + { -461409, -7830014, 20614118, 16688288, -7514766, -4807119, 22300304, 505429, 6108462, -6183415 }, + { -5070281, 12367917, -30663534, 3234473, 32617080, -8422642, 29880583, -13483331, -26898490, -7867459 }, + { -31975283, 5726539, 26934134, 10237677, -3173717, -605053, 24199304, 3795095, 7592688, -14992079 } + }, + { + { 21594432, -14964228, 17466408, -4077222, 32537084, 2739898, 6407723, 12018833, -28256052, 4298412 }, + { -20650503, -11961496, -27236275, 570498, 3767144, -1717540, 13891942, -1569194, 13717174, 10805743 }, + { -14676630, -15644296, 15287174, 11927123, 24177847, -8175568, -796431, 14860609, -26938930, -5863836 } + } +}, +{ /* 13/31 */ + { + { 12962541, 5311799, -10060768, 11658280, 18855286, -7954201, 13286263, -12808704, -4381056, 9882022 }, + { 18512079, 11319350, -20123124, 15090309, 18818594, 5271736, -22727904, 3666879, -23967430, -3299429 }, + { -6789020, -3146043, 16192429, 13241070, 15898607, -14206114, -10084880, -6661110, -2403099, 5276065 } + }, + { + { 30169808, -5317648, 26306206, -11750859, 27814964, 7069267, 7152851, 3684982, 1449224, 13082861 }, + { 10342826, 3098505, 2119311, 193222, 25702612, 12233820, 23697382, 15056736, -21016438, -8202000 }, + { -33150110, 3261608, 22745853, 7948688, 19370557, -15177665, -26171976, 6482814, -10300080, -11060101 } + }, + { + { 32869458, -5408545, 25609743, 15678670, -10687769, -15471071, 26112421, 2521008, -22664288, 6904815 }, + { 29506923, 4457497, 3377935, -9796444, -30510046, 12935080, 1561737, 3841096, -29003639, -6657642 }, + { 10340844, -6630377, -18656632, -2278430, 12621151, -13339055, 30878497, -11824370, -25584551, 5181966 } + }, + { + { 25940115, -12658025, 17324188, -10307374, -8671468, 15029094, 24396252, -16450922, -2322852, -12388574 }, + { -21765684, 9916823, -1300409, 4079498, -1028346, 11909559, 1782390, 12641087, 20603771, -6561742 }, + { -18882287, -11673380, 24849422, 11501709, 13161720, -4768874, 1925523, 11914390, 4662781, 7820689 } + }, + { + { 12241050, -425982, 8132691, 9393934, 32846760, -1599620, 29749456, 12172924, 16136752, 15264020 }, + { -10349955, -14680563, -8211979, 2330220, -17662549, -14545780, 10658213, 6671822, 19012087, 3772772 }, + { 3753511, -3421066, 10617074, 2028709, 14841030, -6721664, 28718732, -15762884, 20527771, 12988982 } + }, + { + { -14822485, -5797269, -3707987, 12689773, -898983, -10914866, -24183046, -10564943, 3299665, -12424953 }, + { -16777703, -15253301, -9642417, 4978983, 3308785, 8755439, 6943197, 6461331, -25583147, 8991218 }, + { -17226263, 1816362, -1673288, -6086439, 31783888, -8175991, -32948145, 7417950, -30242287, 1507265 } + }, + { + { 29692663, 6829891, -10498800, 4334896, 20945975, -11906496, -28887608, 8209391, 14606362, -10647073 }, + { -3481570, 8707081, 32188102, 5672294, 22096700, 1711240, -33020695, 9761487, 4170404, -2085325 }, + { -11587470, 14855945, -4127778, -1531857, -26649089, 15084046, 22186522, 16002000, -14276837, -8400798 } + }, + { + { -4811456, 13761029, -31703877, -2483919, -3312471, 7869047, -7113572, -9620092, 13240845, 10965870 }, + { -7742563, -8256762, -14768334, -13656260, -23232383, 12387166, 4498947, 14147411, 29514390, 4302863 }, + { -13413405, -12407859, 20757302, -13801832, 14785143, 8976368, -5061276, -2144373, 17846988, -13971927 } + } +}, +{ /* 14/31 */ + { + { -2244452, -754728, -4597030, -1066309, -6247172, 1455299, -21647728, -9214789, -5222701, 12650267 }, + { -9906797, -16070310, 21134160, 12198166, -27064575, 708126, 387813, 13770293, -19134326, 10958663 }, + { 22470984, 12369526, 23446014, -5441109, -21520802, -9698723, -11772496, -11574455, -25083830, 4271862 } + }, + { + { -25169565, -10053642, -19909332, 15361595, -5984358, 2159192, 75375, -4278529, -32526221, 8469673 }, + { 15854970, 4148314, -8893890, 7259002, 11666551, 13824734, -30531198, 2697372, 24154791, -9460943 }, + { 15446137, -15806644, 29759747, 14019369, 30811221, -9610191, -31582008, 12840104, 24913809, 9815020 } + }, + { + { -4709286, -5614269, -31841498, -12288893, -14443537, 10799414, -9103676, 13438769, 18735128, 9466238 }, + { 11933045, 9281483, 5081055, -5183824, -2628162, -4905629, -7727821, -10896103, -22728655, 16199064 }, + { 14576810, 379472, -26786533, -8317236, -29426508, -10812974, -102766, 1876699, 30801119, 2164795 } + }, + { + { 15995086, 3199873, 13672555, 13712240, -19378835, -4647646, -13081610, -15496269, -13492807, 1268052 }, + { -10290614, -3659039, -3286592, 10948818, 23037027, 3794475, -3470338, -12600221, -17055369, 3565904 }, + { 29210088, -9419337, -5919792, -4952785, 10834811, -13327726, -16512102, -10820713, -27162222, -14030531 } + }, + { + { -13161890, 15508588, 16663704, -8156150, -28349942, 9019123, -29183421, -3769423, 2244111, -14001979 }, + { -5152875, -3800936, -9306475, -6071583, 16243069, 14684434, -25673088, -16180800, 13491506, 4641841 }, + { 10813417, 643330, -19188515, -728916, 30292062, -16600078, 27548447, -7721242, 14476989, -12767431 } + }, + { + { 10292079, 9984945, 6481436, 8279905, -7251514, 7032743, 27282937, -1644259, -27912810, 12651324 }, + { -31185513, -813383, 22271204, 11835308, 10201545, 15351028, 17099662, 3988035, 21721536, -3148940 }, + { 10202177, -6545839, -31373232, -9574638, -32150642, -8119683, -12906320, 3852694, 13216206, 14842320 } + }, + { + { -15815640, -10601066, -6538952, -7258995, -6984659, -6581778, -31500847, 13765824, -27434397, 9900184 }, + { 14465505, -13833331, -32133984, -14738873, -27443187, 12990492, 33046193, 15796406, -7051866, -8040114 }, + { 30924417, -8279620, 6359016, -12816335, 16508377, 9071735, -25488601, 15413635, 9524356, -7018878 } + }, + { + { 12274201, -13175547, 32627641, -1785326, 6736625, 13267305, 5237659, -5109483, 15663516, 4035784 }, + { -2951309, 8903985, 17349946, 601635, -16432815, -4612556, -13732739, -15889334, -22258478, 4659091 }, + { -16916263, -4952973, -30393711, -15158821, 20774812, 15897498, 5736189, 15026997, -2178256, -13455585 } + } +}, +{ /* 15/31 */ + { + { -8858980, -2219056, 28571666, -10155518, -474467, -10105698, -3801496, 278095, 23440562, -290208 }, + { 10226241, -5928702, 15139956, 120818, -14867693, 5218603, 32937275, 11551483, -16571960, -7442864 }, + { 17932739, -12437276, -24039557, 10749060, 11316803, 7535897, 22503767, 5561594, -3646624, 3898661 } + }, + { + { 7749907, -969567, -16339731, -16464, -25018111, 15122143, -1573531, 7152530, 21831162, 1245233 }, + { 26958459, -14658026, 4314586, 8346991, -5677764, 11960072, -32589295, -620035, -30402091, -16716212 }, + { -12165896, 9166947, 33491384, 13673479, 29787085, 13096535, 6280834, 14587357, -22338025, 13987525 } + }, + { + { -24349909, 7778775, 21116000, 15572597, -4833266, -5357778, -4300898, -5124639, -7469781, -2858068 }, + { 9681908, -6737123, -31951644, 13591838, -6883821, 386950, 31622781, 6439245, -14581012, 4091397 }, + { -8426427, 1470727, -28109679, -1596990, 3978627, -5123623, -19622683, 12092163, 29077877, -14741988 } + }, + { + { 5269168, -6859726, -13230211, -8020715, 25932563, 1763552, -5606110, -5505881, -20017847, 2357889 }, + { 32264008, -15407652, -5387735, -1160093, -2091322, -3946900, 23104804, -12869908, 5727338, 189038 }, + { 14609123, -8954470, -6000566, -16622781, -14577387, -7743898, -26745169, 10942115, -25888931, -14884697 } + }, + { + { 20513500, 5557931, -15604613, 7829531, 26413943, -2019404, -21378968, 7471781, 13913677, -5137875 }, + { -25574376, 11967826, 29233242, 12948236, -6754465, 4713227, -8940970, 14059180, 12878652, 8511905 }, + { -25656801, 3393631, -2955415, -7075526, -2250709, 9366908, -30223418, 6812974, 5568676, -3127656 } + }, + { + { 11630004, 12144454, 2116339, 13606037, 27378885, 15676917, -17408753, -13504373, -14395196, 8070818 }, + { 27117696, -10007378, -31282771, -5570088, 1127282, 12772488, -29845906, 10483306, -11552749, -1028714 }, + { 10637467, -5688064, 5674781, 1072708, -26343588, -6982302, -1683975, 9177853, -27493162, 15431203 } + }, + { + { 20525145, 10892566, -12742472, 12779443, -29493034, 16150075, -28240519, 14943142, -15056790, -7935931 }, + { -30024462, 5626926, -551567, -9981087, 753598, 11981191, 25244767, -3239766, -3356550, 9594024 }, + { -23752644, 2636870, -5163910, -10103818, 585134, 7877383, 11345683, -6492290, 13352335, -10977084 } + }, + { + { -1931799, -5407458, 3304649, -12884869, 17015806, -4877091, -29783850, -7752482, -13215537, -319204 }, + { 20239939, 6607058, 6203985, 3483793, -18386976, -779229, -20723742, 15077870, -22750759, 14523817 }, + { 27406042, -6041657, 27423596, -4497394, 4996214, 10002360, -28842031, -4545494, -30172742, -4805667 } + } +}, +{ /* 16/31 */ + { + { 11374242, 12660715, 17861383, -12540833, 10935568, 1099227, -13886076, -9091740, -27727044, 11358504 }, + { -12730809, 10311867, 1510375, 10778093, -2119455, -9145702, 32676003, 11149336, -26123651, 4985768 }, + { -19096303, 341147, -6197485, -239033, 15756973, -8796662, -983043, 13794114, -19414307, -15621255 } + }, + { + { 6490081, 11940286, 25495923, -7726360, 8668373, -8751316, 3367603, 6970005, -1691065, -9004790 }, + { 1656497, 13457317, 15370807, 6364910, 13605745, 8362338, -19174622, -5475723, -16796596, -5031438 }, + { -22273315, -13524424, -64685, -4334223, -18605636, -10921968, -20571065, -7007978, -99853, -10237333 } + }, + { + { 17747465, 10039260, 19368299, -4050591, -20630635, -16041286, 31992683, -15857976, -29260363, -5511971 }, + { 31932027, -4986141, -19612382, 16366580, 22023614, 88450, 11371999, -3744247, 4882242, -10626905 }, + { 29796507, 37186, 19818052, 10115756, -11829032, 3352736, 18551198, 3272828, -5190932, -4162409 } + }, + { + { 12501286, 4044383, -8612957, -13392385, -32430052, 5136599, -19230378, -3529697, 330070, -3659409 }, + { 6384877, 2899513, 17807477, 7663917, -2358888, 12363165, 25366522, -8573892, -271295, 12071499 }, + { -8365515, -4042521, 25133448, -4517355, -6211027, 2265927, -32769618, 1936675, -5159697, 3829363 } + }, + { + { 28425966, -5835433, -577090, -4697198, -14217555, 6870930, 7921550, -6567787, 26333140, 14267664 }, + { -11067219, 11871231, 27385719, -10559544, -4585914, -11189312, 10004786, -8709488, -21761224, 8930324 }, + { -21197785, -16396035, 25654216, -1725397, 12282012, 11008919, 1541940, 4757911, -26491501, -16408940 } + }, + { + { 13537262, -7759490, -20604840, 10961927, -5922820, -13218065, -13156584, 6217254, -15943699, 13814990 }, + { -17422573, 15157790, 18705543, 29619, 24409717, -260476, 27361681, 9257833, -1956526, -1776914 }, + { -25045300, -10191966, 15366585, 15166509, -13105086, 8423556, -29171540, 12361135, -18685978, 4578290 } + }, + { + { 24579768, 3711570, 1342322, -11180126, -27005135, 14124956, -22544529, 14074919, 21964432, 8235257 }, + { -6528613, -2411497, 9442966, -5925588, 12025640, -1487420, -2981514, -1669206, 13006806, 2355433 }, + { -16304899, -13605259, -6632427, -5142349, 16974359, -10911083, 27202044, 1719366, 1141648, -12796236 } + }, + { + { -12863944, -13219986, -8318266, -11018091, -6810145, -4843894, 13475066, -3133972, 32674895, 13715045 }, + { 11423335, -5468059, 32344216, 8962751, 24989809, 9241752, -13265253, 16086212, -28740881, -15642093 }, + { -1409668, 12530728, -6368726, 10847387, 19531186, -14132160, -11709148, 7791794, -27245943, 4383347 } + } +}, +{ /* 17/31 */ + { + { -28970898, 5271447, -1266009, -9736989, -12455236, 16732599, -4862407, -4906449, 27193557, 6245191 }, + { -15193956, 5362278, -1783893, 2695834, 4960227, 12840725, 23061898, 3260492, 22510453, 8577507 }, + { -12632451, 11257346, -32692994, 13548177, -721004, 10879011, 31168030, 13952092, -29571492, -3635906 } + }, + { + { 3877321, -9572739, 32416692, 5405324, -11004407, -13656635, 3759769, 11935320, 5611860, 8164018 }, + { -16275802, 14667797, 15906460, 12155291, -22111149, -9039718, 32003002, -8832289, 5773085, -8422109 }, + { -23788118, -8254300, 1950875, 8937633, 18686727, 16459170, -905725, 12376320, 31632953, 190926 } + }, + { + { -24593607, -16138885, -8423991, 13378746, 14162407, 6901328, -8288749, 4508564, -25341555, -3627528 }, + { 8884438, -5884009, 6023974, 10104341, -6881569, -4941533, 18722941, -14786005, -1672488, 827625 }, + { -32720583, -16289296, -32503547, 7101210, 13354605, 2659080, -1800575, -14108036, -24878478, 1541286 } + }, + { + { 2901347, -1117687, 3880376, -10059388, -17620940, -3612781, -21802117, -3567481, 20456845, -1885033 }, + { 27019610, 12299467, -13658288, -1603234, -12861660, -4861471, -19540150, -5016058, 29439641, 15138866 }, + { 21536104, -6626420, -32447818, -10690208, -22408077, 5175814, -5420040, -16361163, 7779328, 109896 } + }, + { + { 30279744, 14648750, -8044871, 6425558, 13639621, -743509, 28698390, 12180118, 23177719, -554075 }, + { 26572847, 3405927, -31701700, 12890905, -19265668, 5335866, -6493768, 2378492, 4439158, -13279347 }, + { -22716706, 3489070, -9225266, -332753, 18875722, -1140095, 14819434, -12731527, -17717757, -5461437 } + }, + { + { -5056483, 16566551, 15953661, 3767752, -10436499, 15627060, -820954, 2177225, 8550082, -15114165 }, + { -18473302, 16596775, -381660, 15663611, 22860960, 15585581, -27844109, -3582739, -23260460, -8428588 }, + { -32480551, 15707275, -8205912, -5652081, 29464558, 2713815, -22725137, 15860482, -21902570, 1494193 } + }, + { + { -19562091, -14087393, -25583872, -9299552, 13127842, 759709, 21923482, 16529112, 8742704, 12967017 }, + { -28464899, 1553205, 32536856, -10473729, -24691605, -406174, -8914625, -2933896, -29903758, 15553883 }, + { 21877909, 3230008, 9881174, 10539357, -4797115, 2841332, 11543572, 14513274, 19375923, -12647961 } + }, + { + { 8832269, -14495485, 13253511, 5137575, 5037871, 4078777, 24880818, -6222716, 2862653, 9455043 }, + { 29306751, 5123106, 20245049, -14149889, 9592566, 8447059, -2077124, -2990080, 15511449, 4789663 }, + { -20679756, 7004547, 8824831, -9434977, -4045704, -3750736, -5754762, 108893, 23513200, 16652362 } + } +}, +{ /* 18/31 */ + { + { -33256173, 4144782, -4476029, -6579123, 10770039, -7155542, -6650416, -12936300, -18319198, 10212860 }, + { 2756081, 8598110, 7383731, -6859892, 22312759, -1105012, 21179801, 2600940, -9988298, -12506466 }, + { -24645692, 13317462, -30449259, -15653928, 21365574, -10869657, 11344424, 864440, -2499677, -16710063 } + }, + { + { -26432803, 6148329, -17184412, -14474154, 18782929, -275997, -22561534, 211300, 2719757, 4940997 }, + { -1323882, 3911313, -6948744, 14759765, -30027150, 7851207, 21690126, 8518463, 26699843, 5276295 }, + { -13149873, -6429067, 9396249, 365013, 24703301, -10488939, 1321586, 149635, -15452774, 7159369 } + }, + { + { 9987780, -3404759, 17507962, 9505530, 9731535, -2165514, 22356009, 8312176, 22477218, -8403385 }, + { 18155857, -16504990, 19744716, 9006923, 15154154, -10538976, 24256460, -4864995, -22548173, 9334109 }, + { 2986088, -4911893, 10776628, -3473844, 10620590, -7083203, -21413845, 14253545, -22587149, 536906 } + }, + { + { 4377756, 8115836, 24567078, 15495314, 11625074, 13064599, 7390551, 10589625, 10838060, -15420424 }, + { -19342404, 867880, 9277171, -3218459, -14431572, -1986443, 19295826, -15796950, 6378260, 699185 }, + { 7895026, 4057113, -7081772, -13077756, -17886831, -323126, -716039, 15693155, -5045064, -13373962 } + }, + { + { -7737563, -5869402, -14566319, -7406919, 11385654, 13201616, 31730678, -10962840, -3918636, -9669325 }, + { 10188286, -15770834, -7336361, 13427543, 22223443, 14896287, 30743455, 7116568, -21786507, 5427593 }, + { 696102, 13206899, 27047647, -10632082, 15285305, -9853179, 10798490, -4578720, 19236243, 12477404 } + }, + { + { -11229439, 11243796, -17054270, -8040865, -788228, -8167967, -3897669, 11180504, -23169516, 7733644 }, + { 17800790, -14036179, -27000429, -11766671, 23887827, 3149671, 23466177, -10538171, 10322027, 15313801 }, + { 26246234, 11968874, 32263343, -5468728, 6830755, -13323031, -15794704, -101982, -24449242, 10890804 } + }, + { + { -31365647, 10271363, -12660625, -6267268, 16690207, -13062544, -14982212, 16484931, 25180797, -5334884 }, + { -586574, 10376444, -32586414, -11286356, 19801893, 10997610, 2276632, 9482883, 316878, 13820577 }, + { -9882808, -4510367, -2115506, 16457136, -11100081, 11674996, 30756178, -7515054, 30696930, -3712849 } + }, + { + { 32988917, -9603412, 12499366, 7910787, -10617257, -11931514, -7342816, -9985397, -32349517, 7392473 }, + { -8855661, 15927861, 9866406, -3649411, -2396914, -16655781, -30409476, -9134995, 25112947, -2926644 }, + { -2504044, -436966, 25621774, -5678772, 15085042, -5479877, -24884878, -13526194, 5537438, -13914319 } + } +}, +{ /* 19/31 */ + { + { -11225584, 2320285, -9584280, 10149187, -33444663, 5808648, -14876251, -1729667, 31234590, 6090599 }, + { -9633316, 116426, 26083934, 2897444, -6364437, -2688086, 609721, 15878753, -6970405, -9034768 }, + { -27757857, 247744, -15194774, -9002551, 23288161, -10011936, -23869595, 6503646, 20650474, 1804084 } + }, + { + { -27589786, 15456424, 8972517, 8469608, 15640622, 4439847, 3121995, -10329713, 27842616, -202328 }, + { -15306973, 2839644, 22530074, 10026331, 4602058, 5048462, 28248656, 5031932, -11375082, 12714369 }, + { 20807691, -7270825, 29286141, 11421711, -27876523, -13868230, -21227475, 1035546, -19733229, 12796920 } + }, + { + { 12076899, -14301286, -8785001, -11848922, -25012791, 16400684, -17591495, -12899438, 3480665, -15182815 }, + { -32361549, 5457597, 28548107, 7833186, 7303070, -11953545, -24363064, -15921875, -33374054, 2771025 }, + { -21389266, 421932, 26597266, 6860826, 22486084, -6737172, -17137485, -4210226, -24552282, 15673397 } + }, + { + { -20184622, 2338216, 19788685, -9620956, -4001265, -8740893, -20271184, 4733254, 3727144, -12934448 }, + { 6120119, 814863, -11794402, -622716, 6812205, -15747771, 2019594, 7975683, 31123697, -10958981 }, + { 30069250, -11435332, 30434654, 2958439, 18399564, -976289, 12296869, 9204260, -16432438, 9648165 } + }, + { + { 32705432, -1550977, 30705658, 7451065, -11805606, 9631813, 3305266, 5248604, -26008332, -11377501 }, + { 17219865, 2375039, -31570947, -5575615, -19459679, 9219903, 294711, 15298639, 2662509, -16297073 }, + { -1172927, -7558695, -4366770, -4287744, -21346413, -8434326, 32087529, -1222777, 32247248, -14389861 } + }, + { + { 14312628, 1221556, 17395390, -8700143, -4945741, -8684635, -28197744, -9637817, -16027623, -13378845 }, + { -1428825, -9678990, -9235681, 6549687, -7383069, -468664, 23046502, 9803137, 17597934, 2346211 }, + { 18510800, 15337574, 26171504, 981392, -22241552, 7827556, -23491134, -11323352, 3059833, -11782870 } + }, + { + { 10141598, 6082907, 17829293, -1947643, 9830092, 13613136, -25556636, -5544586, -33502212, 3592096 }, + { 33114168, -15889352, -26525686, -13343397, 33076705, 8716171, 1151462, 1521897, -982665, -6837803 }, + { -32939165, -4255815, 23947181, -324178, -33072974, -12305637, -16637686, 3891704, 26353178, 693168 } + }, + { + { 30374239, 1595580, -16884039, 13186931, 4600344, 406904, 9585294, -400668, 31375464, 14369965 }, + { -14370654, -7772529, 1510301, 6434173, -18784789, -6262728, 32732230, -13108839, 17901441, 16011505 }, + { 18171223, -11934626, -12500402, 15197122, -11038147, -15230035, -19172240, -16046376, 8764035, 12309598 } + } +}, +{ /* 20/31 */ + { + { 5975908, -5243188, -19459362, -9681747, -11541277, 14015782, -23665757, 1228319, 17544096, -10593782 }, + { 5811932, -1715293, 3442887, -2269310, -18367348, -8359541, -18044043, -15410127, -5565381, 12348900 }, + { -31399660, 11407555, 25755363, 6891399, -3256938, 14872274, -24849353, 8141295, -10632534, -585479 } + }, + { + { -12675304, 694026, -5076145, 13300344, 14015258, -14451394, -9698672, -11329050, 30944593, 1130208 }, + { 8247766, -6710942, -26562381, -7709309, -14401939, -14648910, 4652152, 2488540, 23550156, -271232 }, + { 17294316, -3788438, 7026748, 15626851, 22990044, 113481, 2267737, -5908146, -408818, -137719 } + }, + { + { 16091085, -16253926, 18599252, 7340678, 2137637, -1221657, -3364161, 14550936, 3260525, -7166271 }, + { -4910104, -13332887, 18550887, 10864893, -16459325, -7291596, -23028869, -13204905, -12748722, 2701326 }, + { -8574695, 16099415, 4629974, -16340524, -20786213, -6005432, -10018363, 9276971, 11329923, 1862132 } + }, + { + { 14763076, -15903608, -30918270, 3689867, 3511892, 10313526, -21951088, 12219231, -9037963, -940300 }, + { 8894987, -3446094, 6150753, 3013931, 301220, 15693451, -31981216, -2909717, -15438168, 11595570 }, + { 15214962, 3537601, -26238722, -14058872, 4418657, -15230761, 13947276, 10730794, -13489462, -4363670 } + }, + { + { -2538306, 7682793, 32759013, 263109, -29984731, -7955452, -22332124, -10188635, 977108, 699994 }, + { -12466472, 4195084, -9211532, 550904, -15565337, 12917920, 19118110, -439841, -30534533, -14337913 }, + { 31788461, -14507657, 4799989, 7372237, 8808585, -14747943, 9408237, -10051775, 12493932, -5409317 } + }, + { + { -25680606, 5260744, -19235809, -6284470, -3695942, 16566087, 27218280, 2607121, 29375955, 6024730 }, + { 842132, -2794693, -4763381, -8722815, 26332018, -12405641, 11831880, 6985184, -9940361, 2854096 }, + { -4847262, -7969331, 2516242, -5847713, 9695691, -7221186, 16512645, 960770, 12121869, 16648078 } + }, + { + { -15218652, 14667096, -13336229, 2013717, 30598287, -464137, -31504922, -7882064, 20237806, 2838411 }, + { -19288047, 4453152, 15298546, -16178388, 22115043, -15972604, 12544294, -13470457, 1068881, -12499905 }, + { -9558883, -16518835, 33238498, 13506958, 30505848, -1114596, -8486907, -2630053, 12521378, 4845654 } + }, + { + { -28198521, 10744108, -2958380, 10199664, 7759311, -13088600, 3409348, -873400, -6482306, -12885870 }, + { -23561822, 6230156, -20382013, 10655314, -24040585, -11621172, 10477734, -1240216, -3113227, 13974498 }, + { 12966261, 15550616, -32038948, -1615346, 21025980, -629444, 5642325, 7188737, 18895762, 12629579 } + } +}, +{ /* 21/31 */ + { + { 14741879, -14946887, 22177208, -11721237, 1279741, 8058600, 11758140, 789443, 32195181, 3895677 }, + { 10758205, 15755439, -4509950, 9243698, -4879422, 6879879, -2204575, -3566119, -8982069, 4429647 }, + { -2453894, 15725973, -20436342, -10410672, -5803908, -11040220, -7135870, -11642895, 18047436, -15281743 } + }, + { + { -25173001, -11307165, 29759956, 11776784, -22262383, -15820455, 10993114, -12850837, -17620701, -9408468 }, + { 21987233, 700364, -24505048, 14972008, -7774265, -5718395, 32155026, 2581431, -29958985, 8773375 }, + { -25568350, 454463, -13211935, 16126715, 25240068, 8594567, 20656846, 12017935, -7874389, -13920155 } + }, + { + { 6028182, 6263078, -31011806, -11301710, -818919, 2461772, -31841174, -5468042, -1721788, -2776725 }, + { -12278994, 16624277, 987579, -5922598, 32908203, 1248608, 7719845, -4166698, 28408820, 6816612 }, + { -10358094, -8237829, 19549651, -12169222, 22082623, 16147817, 20613181, 13982702, -10339570, 5067943 } + }, + { + { -30505967, -3821767, 12074681, 13582412, -19877972, 2443951, -19719286, 12746132, 5331210, -10105944 }, + { 30528811, 3601899, -1957090, 4619785, -27361822, -15436388, 24180793, -12570394, 27679908, -1648928 }, + { 9402404, -13957065, 32834043, 10838634, -26580150, -13237195, 26653274, -8685565, 22611444, -12715406 } + }, + { + { 22190590, 1118029, 22736441, 15130463, -30460692, -5991321, 19189625, -4648942, 4854859, 6622139 }, + { -8310738, -2953450, -8262579, -3388049, -10401731, -271929, 13424426, -3567227, 26404409, 13001963 }, + { -31241838, -15415700, -2994250, 8939346, 11562230, -12840670, -26064365, -11621720, -15405155, 11020693 } + }, + { + { 1866042, -7949489, -7898649, -10301010, 12483315, 13477547, 3175636, -12424163, 28761762, 1406734 }, + { -448555, -1777666, 13018551, 3194501, -9580420, -11161737, 24760585, -4347088, 25577411, -13378680 }, + { -24290378, 4759345, -690653, -1852816, 2066747, 10693769, -29595790, 9884936, -9368926, 4745410 } + }, + { + { -9141284, 6049714, -19531061, -4341411, -31260798, 9944276, -15462008, -11311852, 10931924, -11931931 }, + { -16561513, 14112680, -8012645, 4817318, -8040464, -11414606, -22853429, 10856641, -20470770, 13434654 }, + { 22759489, -10073434, -16766264, -1871422, 13637442, -10168091, 1765144, -12654326, 28445307, -5364710 } + }, + { + { 29875063, 12493613, 2795536, -3786330, 1710620, 15181182, -10195717, -8788675, 9074234, 1167180 }, + { -26205683, 11014233, -9842651, -2635485, -26908120, 7532294, -18716888, -9535498, 3843903, 9367684 }, + { -10969595, -6403711, 9591134, 9582310, 11349256, 108879, 16235123, 8601684, -139197, 4242895 } + } +}, +{ /* 22/31 */ + { + { 22092954, -13191123, -2042793, -11968512, 32186753, -11517388, -6574341, 2470660, -27417366, 16625501 }, + { -11057722, 3042016, 13770083, -9257922, 584236, -544855, -7770857, 2602725, -27351616, 14247413 }, + { 6314175, -10264892, -32772502, 15957557, -10157730, 168750, -8618807, 14290061, 27108877, -1180880 } + }, + { + { -8586597, -7170966, 13241782, 10960156, -32991015, -13794596, 33547976, -11058889, -27148451, 981874 }, + { 22833440, 9293594, -32649448, -13618667, -9136966, 14756819, -22928859, -13970780, -10479804, -16197962 }, + { -7768587, 3326786, -28111797, 10783824, 19178761, 14905060, 22680049, 13906969, -15933690, 3797899 } + }, + { + { 21721356, -4212746, -12206123, 9310182, -3882239, -13653110, 23740224, -2709232, 20491983, -8042152 }, + { 9209270, -15135055, -13256557, -6167798, -731016, 15289673, 25947805, 15286587, 30997318, -6703063 }, + { 7392032, 16618386, 23946583, -8039892, -13265164, -1533858, -14197445, -2321576, 17649998, -250080 } + }, + { + { -9301088, -14193827, 30609526, -3049543, -25175069, -1283752, -15241566, -9525724, -2233253, 7662146 }, + { -17558673, 1763594, -33114336, 15908610, -30040870, -12174295, 7335080, -8472199, -3174674, 3440183 }, + { -19889700, -5977008, -24111293, -9688870, 10799743, -16571957, 40450, -4431835, 4862400, 1133 } + }, + { + { -32856209, -7873957, -5422389, 14860950, -16319031, 7956142, 7258061, 311861, -30594991, -7379421 }, + { -3773428, -1565936, 28985340, 7499440, 24445838, 9325937, 29727763, 16527196, 18278453, 15405622 }, + { -4381906, 8508652, -19898366, -3674424, -5984453, 15149970, -13313598, 843523, -21875062, 13626197 } + }, + { + { 2281448, -13487055, -10915418, -2609910, 1879358, 16164207, -10783882, 3953792, 13340839, 15928663 }, + { 31727126, -7179855, -18437503, -8283652, 2875793, -16390330, -25269894, -7014826, -23452306, 5964753 }, + { 4100420, -5959452, -17179337, 6017714, -18705837, 12227141, -26684835, 11344144, 2538215, -7570755 } + }, + { + { -9433605, 6123113, 11159803, -2156608, 30016280, 14966241, -20474983, 1485421, -629256, -15958862 }, + { -26804558, 4260919, 11851389, 9658551, -32017107, 16367492, -20205425, -13191288, 11659922, -11115118 }, + { 26180396, 10015009, -30844224, -8581293, 5418197, 9480663, 2231568, -10170080, 33100372, -1306171 } + }, + { + { 15121113, -5201871, -10389905, 15427821, -27509937, -15992507, 21670947, 4486675, -5931810, -14466380 }, + { 16166486, -9483733, -11104130, 6023908, -31926798, -1364923, 2340060, -16254968, -10735770, -10039824 }, + { 28042865, -3557089, -12126526, 12259706, -3717498, -6945899, 6766453, -8689599, 18036436, 5803270 } + } +}, +{ /* 23/31 */ + { + { -817581, 6763912, 11803561, 1585585, 10958447, -2671165, 23855391, 4598332, -6159431, -14117438 }, + { -31031306, -14256194, 17332029, -2383520, 31312682, -5967183, 696309, 50292, -20095739, 11763584 }, + { -594563, -2514283, -32234153, 12643980, 12650761, 14811489, 665117, -12613632, -19773211, -10713562 } + }, + { + { 30464590, -11262872, -4127476, -12734478, 19835327, -7105613, -24396175, 2075773, -17020157, 992471 }, + { 18357185, -6994433, 7766382, 16342475, -29324918, 411174, 14578841, 8080033, -11574335, -10601610 }, + { 19598397, 10334610, 12555054, 2555664, 18821899, -10339780, 21873263, 16014234, 26224780, 16452269 } + }, + { + { -30223925, 5145196, 5944548, 16385966, 3976735, 2009897, -11377804, -7618186, -20533829, 3698650 }, + { 14187449, 3448569, -10636236, -10810935, -22663880, -3433596, 7268410, -10890444, 27394301, 12015369 }, + { 19695761, 16087646, 28032085, 12999827, 6817792, 11427614, 20244189, -1312777, -13259127, -3402461 } + }, + { + { 30860103, 12735208, -1888245, -4699734, -16974906, 2256940, -8166013, 12298312, -8550524, -10393462 }, + { -5719826, -11245325, -1910649, 15569035, 26642876, -7587760, -5789354, -15118654, -4976164, 12651793 }, + { -2848395, 9953421, 11531313, -5282879, 26895123, -12697089, -13118820, -16517902, 9768698, -2533218 } + }, + { + { -24719459, 1894651, -287698, -4704085, 15348719, -8156530, 32767513, 12765450, 4940095, 10678226 }, + { 18860224, 15980149, -18987240, -1562570, -26233012, -11071856, -7843882, 13944024, -24372348, 16582019 }, + { -15504260, 4970268, -29893044, 4175593, -20993212, -2199756, -11704054, 15444560, -11003761, 7989037 } + }, + { + { 31490452, 5568061, -2412803, 2182383, -32336847, 4531686, -32078269, 6200206, -19686113, -14800171 }, + { -17308668, -15879940, -31522777, -2831, -32887382, 16375549, 8680158, -16371713, 28550068, -6857132 }, + { -28126887, -5688091, 16837845, -1820458, -6850681, 12700016, -30039981, 4364038, 1155602, 5988841 } + }, + { + { 21890435, -13272907, -12624011, 12154349, -7831873, 15300496, 23148983, -4470481, 24618407, 8283181 }, + { -33136107, -10512751, 9975416, 6841041, -31559793, 16356536, 3070187, -7025928, 1466169, 10740210 }, + { -1509399, -15488185, -13503385, -10655916, 32799044, 909394, -13938903, -5779719, -32164649, -15327040 } + }, + { + { 3960823, -14267803, -28026090, -15918051, -19404858, 13146868, 15567327, 951507, -3260321, -573935 }, + { 24740841, 5052253, -30094131, 8961361, 25877428, 6165135, -24368180, 14397372, -7380369, -6144105 }, + { -28888365, 3510803, -28103278, -1158478, -11238128, -10631454, -15441463, -14453128, -1625486, -6494814 } + } +}, +{ /* 24/31 */ + { + { 793299, -9230478, 8836302, -6235707, -27360908, -2369593, 33152843, -4885251, -9906200, -621852 }, + { 5666233, 525582, 20782575, -8038419, -24538499, 14657740, 16099374, 1468826, -6171428, -15186581 }, + { -4859255, -3779343, -2917758, -6748019, 7778750, 11688288, -30404353, -9871238, -1558923, -9863646 } + }, + { + { 10896332, -7719704, 824275, 472601, -19460308, 3009587, 25248958, 14783338, -30581476, -15757844 }, + { 10566929, 12612572, -31944212, 11118703, -12633376, 12362879, 21752402, 8822496, 24003793, 14264025 }, + { 27713862, -7355973, -11008240, 9227530, 27050101, 2504721, 23886875, -13117525, 13958495, -5732453 } + }, + { + { -23481610, 4867226, -27247128, 3900521, 29838369, -8212291, -31889399, -10041781, 7340521, -15410068 }, + { 4646514, -8011124, -22766023, -11532654, 23184553, 8566613, 31366726, -1381061, -15066784, -10375192 }, + { -17270517, 12723032, -16993061, 14878794, 21619651, -6197576, 27584817, 3093888, -8843694, 3849921 } + }, + { + { -9064912, 2103172, 25561640, -15125738, -5239824, 9582958, 32477045, -9017955, 5002294, -15550259 }, + { -12057553, -11177906, 21115585, -13365155, 8808712, -12030708, 16489530, 13378448, -25845716, 12741426 }, + { -5946367, 10645103, -30911586, 15390284, -3286982, -7118677, 24306472, 15852464, 28834118, -7646072 } + }, + { + { -17335748, -9107057, -24531279, 9434953, -8472084, -583362, -13090771, 455841, 20461858, 5491305 }, + { 13669248, -16095482, -12481974, -10203039, -14569770, -11893198, -24995986, 11293807, -28588204, -9421832 }, + { 28497928, 6272777, -33022994, 14470570, 8906179, -1225630, 18504674, -14165166, 29867745, -8795943 } + }, + { + { -16207023, 13517196, -27799630, -13697798, 24009064, -6373891, -6367600, -13175392, 22853429, -4012011 }, + { 24191378, 16712145, -13931797, 15217831, 14542237, 1646131, 18603514, -11037887, 12876623, -2112447 }, + { 17902668, 4518229, -411702, -2829247, 26878217, 5258055, -12860753, 608397, 16031844, 3723494 } + }, + { + { -28632773, 12763728, -20446446, 7577504, 33001348, -13017745, 17558842, -7872890, 23896954, -4314245 }, + { -20005381, -12011952, 31520464, 605201, 2543521, 5991821, -2945064, 7229064, -9919646, -8826859 }, + { 28816045, 298879, -28165016, -15920938, 19000928, -1665890, -12680833, -2949325, -18051778, -2082915 } + }, + { + { 16000882, -344896, 3493092, -11447198, -29504595, -13159789, 12577740, 16041268, -19715240, 7847707 }, + { 10151868, 10572098, 27312476, 7922682, 14825339, 4723128, -32855931, -6519018, -10020567, 3852848 }, + { -11430470, 15697596, -21121557, -4420647, 5386314, 15063598, 16514493, -15932110, 29330899, -15076224 } + } +}, +{ /* 25/31 */ + { + { -25499735, -4378794, -15222908, -6901211, 16615731, 2051784, 3303702, 15490, -27548796, 12314391 }, + { 15683520, -6003043, 18109120, -9980648, 15337968, -5997823, -16717435, 15921866, 16103996, -3731215 }, + { -23169824, -10781249, 13588192, -1628807, -3798557, -1074929, -19273607, 5402699, -29815713, -9841101 } + }, + { + { 23190676, 2384583, -32714340, 3462154, -29903655, -1529132, -11266856, 8911517, -25205859, 2739713 }, + { 21374101, -3554250, -33524649, 9874411, 15377179, 11831242, -33529904, 6134907, 4931255, 11987849 }, + { -7732, -2978858, -16223486, 7277597, 105524, -322051, -31480539, 13861388, -30076310, 10117930 } + }, + { + { -29501170, -10744872, -26163768, 13051539, -25625564, 5089643, -6325503, 6704079, 12890019, 15728940 }, + { -21972360, -11771379, -951059, -4418840, 14704840, 2695116, 903376, -10428139, 12885167, 8311031 }, + { -17516482, 5352194, 10384213, -13811658, 7506451, 13453191, 26423267, 4384730, 1888765, -5435404 } + }, + { + { -25817338, -3107312, -13494599, -3182506, 30896459, -13921729, -32251644, -12707869, -19464434, -3340243 }, + { -23607977, -2665774, -526091, 4651136, 5765089, 4618330, 6092245, 14845197, 17151279, -9854116 }, + { -24830458, -12733720, -15165978, 10367250, -29530908, -265356, 22825805, -7087279, -16866484, 16176525 } + }, + { + { -23583256, 6564961, 20063689, 3798228, -4740178, 7359225, 2006182, -10363426, -28746253, -10197509 }, + { -10626600, -4486402, -13320562, -5125317, 3432136, -6393229, 23632037, -1940610, 32808310, 1099883 }, + { 15030977, 5768825, -27451236, -2887299, -6427378, -15361371, -15277896, -6809350, 2051441, -15225865 } + }, + { + { -3362323, -7239372, 7517890, 9824992, 23555850, 295369, 5148398, -14154188, -22686354, 16633660 }, + { 4577086, -16752288, 13249841, -15304328, 19958763, -14537274, 18559670, -10759549, 8402478, -9864273 }, + { -28406330, -1051581, -26790155, -907698, -17212414, -11030789, 9453451, -14980072, 17983010, 9967138 } + }, + { + { -25762494, 6524722, 26585488, 9969270, 24709298, 1220360, -1677990, 7806337, 17507396, 3651560 }, + { -10420457, -4118111, 14584639, 15971087, -15768321, 8861010, 26556809, -5574557, -18553322, -11357135 }, + { 2839101, 14284142, 4029895, 3472686, 14402957, 12689363, -26642121, 8459447, -5605463, -7621941 } + }, + { + { -4839289, -3535444, 9744961, 2871048, 25113978, 3187018, -25110813, -849066, 17258084, -7977739 }, + { 18164541, -10595176, -17154882, -1542417, 19237078, -9745295, 23357533, -15217008, 26908270, 12150756 }, + { -30264870, -7647865, 5112249, -7036672, -1499807, -6974257, 43168, -5537701, -32302074, 16215819 } + } +}, +{ /* 26/31 */ + { + { -6898905, 9824394, -12304779, -4401089, -31397141, -6276835, 32574489, 12532905, -7503072, -8675347 }, + { -27343522, -16515468, -27151524, -10722951, 946346, 16291093, 254968, 7168080, 21676107, -1943028 }, + { 21260961, -8424752, -16831886, -11920822, -23677961, 3968121, -3651949, -6215466, -3556191, -7913075 } + }, + { + { 16544754, 13250366, -16804428, 15546242, -4583003, 12757258, -2462308, -8680336, -18907032, -9662799 }, + { -2415239, -15577728, 18312303, 4964443, -15272530, -12653564, 26820651, 16690659, 25459437, -4564609 }, + { -25144690, 11425020, 28423002, -11020557, -6144921, -15826224, 9142795, -2391602, -6432418, -1644817 } + }, + { + { -23104652, 6253476, 16964147, -3768872, -25113972, -12296437, -27457225, -16344658, 6335692, 7249989 }, + { -30333227, 13979675, 7503222, -12368314, -11956721, -4621693, -30272269, 2682242, 25993170, -12478523 }, + { 4364628, 5930691, 32304656, -10044554, -8054781, 15091131, 22857016, -10598955, 31820368, 15075278 } + }, + { + { 31879134, -8918693, 17258761, 90626, -8041836, -4917709, 24162788, -9650886, -17970238, 12833045 }, + { 19073683, 14851414, -24403169, -11860168, 7625278, 11091125, -19619190, 2074449, -9413939, 14905377 }, + { 24483667, -11935567, -2518866, -11547418, -1553130, 15355506, -25282080, 9253129, 27628530, -7555480 } + }, + { + { 17597607, 8340603, 19355617, 552187, 26198470, -3176583, 4593324, -9157582, -14110875, 15297016 }, + { 510886, 14337390, -31785257, 16638632, 6328095, 2713355, -20217417, -11864220, 8683221, 2921426 }, + { 18606791, 11874196, 27155355, -5281482, -24031742, 6265446, -25178240, -1278924, 4674690, 13890525 } + }, + { + { 13609624, 13069022, -27372361, -13055908, 24360586, 9592974, 14977157, 9835105, 4389687, 288396 }, + { 9922506, -519394, 13613107, 5883594, -18758345, -434263, -12304062, 8317628, 23388070, 16052080 }, + { 12720016, 11937594, -31970060, -5028689, 26900120, 8561328, -20155687, -11632979, -14754271, -10812892 } + }, + { + { 15961858, 14150409, 26716931, -665832, -22794328, 13603569, 11829573, 7467844, -28822128, 929275 }, + { 11038231, -11582396, -27310482, -7316562, -10498527, -16307831, -23479533, -9371869, -21393143, 2465074 }, + { 20017163, -4323226, 27915242, 1529148, 12396362, 15675764, 13817261, -9658066, 2463391, -4622140 } + }, + { + { -16358878, -12663911, -12065183, 4996454, -1256422, 1073572, 9583558, 12851107, 4003896, 12673717 }, + { -1731589, -15155870, -3262930, 16143082, 19294135, 13385325, 14741514, -9103726, 7903886, 2348101 }, + { 24536016, -16515207, 12715592, -3862155, 1511293, 10047386, -3842346, -7129159, -28377538, 10048127 } + } +}, +{ /* 27/31 */ + { + { -12622226, -6204820, 30718825, 2591312, -10617028, 12192840, 18873298, -7297090, -32297756, 15221632 }, + { -26478122, -11103864, 11546244, -1852483, 9180880, 7656409, -21343950, 2095755, 29769758, 6593415 }, + { -31994208, -2907461, 4176912, 3264766, 12538965, -868111, 26312345, -6118678, 30958054, 8292160 } + }, + { + { 31429822, -13959116, 29173532, 15632448, 12174511, -2760094, 32808831, 3977186, 26143136, -3148876 }, + { 22648901, 1402143, -22799984, 13746059, 7936347, 365344, -8668633, -1674433, -3758243, -2304625 }, + { -15491917, 8012313, -2514730, -12702462, -23965846, -10254029, -1612713, -1535569, -16664475, 8194478 } + }, + { + { 27338066, -7507420, -7414224, 10140405, -19026427, -6589889, 27277191, 8855376, 28572286, 3005164 }, + { 26287124, 4821776, 25476601, -4145903, -3764513, -15788984, -18008582, 1182479, -26094821, -13079595 }, + { -7171154, 3178080, 23970071, 6201893, -17195577, -4489192, -21876275, -13982627, 32208683, -1198248 } + }, + { + { -16657702, 2817643, -10286362, 14811298, 6024667, 13349505, -27315504, -10497842, -27672585, -11539858 }, + { 15941029, -9405932, -21367050, 8062055, 31876073, -238629, -15278393, -1444429, 15397331, -4130193 }, + { 8934485, -13485467, -23286397, -13423241, -32446090, 14047986, 31170398, -1441021, -27505566, 15087184 } + }, + { + { -18357243, -2156491, 24524913, -16677868, 15520427, -6360776, -15502406, 11461896, 16788528, -5868942 }, + { -1947386, 16013773, 21750665, 3714552, -17401782, -16055433, -3770287, -10323320, 31322514, -11615635 }, + { 21426655, -5650218, -13648287, -5347537, -28812189, -4920970, -18275391, -14621414, 13040862, -12112948 } + }, + { + { 11293895, 12478086, -27136401, 15083750, -29307421, 14748872, 14555558, -13417103, 1613711, 4896935 }, + { -25894883, 15323294, -8489791, -8057900, 25967126, -13425460, 2825960, -4897045, -23971776, -11267415 }, + { -15924766, -5229880, -17443532, 6410664, 3622847, 10243618, 20615400, 12405433, -23753030, -8436416 } + }, + { + { -7091295, 12556208, -20191352, 9025187, -17072479, 4333801, 4378436, 2432030, 23097949, -566018 }, + { 4565804, -16025654, 20084412, -7842817, 1724999, 189254, 24767264, 10103221, -18512313, 2424778 }, + { 366633, -11976806, 8173090, -6890119, 30788634, 5745705, -7168678, 1344109, -3642553, 12412659 } + }, + { + { -24001791, 7690286, 14929416, -168257, -32210835, -13412986, 24162697, -15326504, -3141501, 11179385 }, + { 18289522, -14724954, 8056945, 16430056, -21729724, 7842514, -6001441, -1486897, -18684645, -11443503 }, + { 476239, 6601091, -6152790, -9723375, 17503545, -4863900, 27672959, 13403813, 11052904, 5219329 } + } +}, +{ /* 28/31 */ + { + { 20678546, -8375738, -32671898, 8849123, -5009758, 14574752, 31186971, -3973730, 9014762, -8579056 }, + { -13644050, -10350239, -15962508, 5075808, -1514661, -11534600, -33102500, 9160280, 8473550, -3256838 }, + { 24900749, 14435722, 17209120, -15292541, -22592275, 9878983, -7689309, -16335821, -24568481, 11788948 } + }, + { + { -3118155, -11395194, -13802089, 14797441, 9652448, -6845904, -20037437, 10410733, -24568470, -1458691 }, + { -15659161, 16736706, -22467150, 10215878, -9097177, 7563911, 11871841, -12505194, -18513325, 8464118 }, + { -23400612, 8348507, -14585951, -861714, -3950205, -6373419, 14325289, 8628612, 33313881, -8370517 } + }, + { + { -20186973, -4967935, 22367356, 5271547, -1097117, -4788838, -24805667, -10236854, -8940735, -5818269 }, + { -6948785, -1795212, -32625683, -16021179, 32635414, -7374245, 15989197, -12838188, 28358192, -4253904 }, + { -23561781, -2799059, -32351682, -1661963, -9147719, 10429267, -16637684, 4072016, -5351664, 5596589 } + }, + { + { -28236598, -3390048, 12312896, 6213178, 3117142, 16078565, 29266239, 2557221, 1768301, 15373193 }, + { -7243358, -3246960, -4593467, -7553353, -127927, -912245, -1090902, -4504991, -24660491, 3442910 }, + { -30210571, 5124043, 14181784, 8197961, 18964734, -11939093, 22597931, 7176455, -18585478, 13365930 } + }, + { + { -7877390, -1499958, 8324673, 4690079, 6261860, 890446, 24538107, -8570186, -9689599, -3031667 }, + { 25008904, -10771599, -4305031, -9638010, 16265036, 15721635, 683793, -11823784, 15723479, -15163481 }, + { -9660625, 12374379, -27006999, -7026148, -7724114, -12314514, 11879682, 5400171, 519526, -1235876 } + }, + { + { 22258397, -16332233, -7869817, 14613016, -22520255, -2950923, -20353881, 7315967, 16648397, 7605640 }, + { -8081308, -8464597, -8223311, 9719710, 19259459, -15348212, 23994942, -5281555, -9468848, 4763278 }, + { -21699244, 9220969, -15730624, 1084137, -25476107, -2852390, 31088447, -7764523, -11356529, 728112 } + }, + { + { 26047220, -11751471, -6900323, -16521798, 24092068, 9158119, -4273545, -12555558, -29365436, -5498272 }, + { 17510331, -322857, 5854289, 8403524, 17133918, -3112612, -28111007, 12327945, 10750447, 10014012 }, + { -10312768, 3936952, 9156313, -8897683, 16498692, -994647, -27481051, -666732, 3424691, 7540221 } + }, + { + { 30322361, -6964110, 11361005, -4143317, 7433304, 4989748, -7071422, -16317219, -9244265, 15258046 }, + { 13054562, -2779497, 19155474, 469045, -12482797, 4566042, 5631406, 2711395, 1062915, -5136345 }, + { -19240248, -11254599, -29509029, -7499965, -5835763, 13005411, -6066489, 12194497, 32960380, 1459310 } + } +}, +{ /* 29/31 */ + { + { 19852034, 7027924, 23669353, 10020366, 8586503, -6657907, 394197, -6101885, 18638003, -11174937 }, + { 31395534, 15098109, 26581030, 8030562, -16527914, -5007134, 9012486, -7584354, -6643087, -5442636 }, + { -9192165, -2347377, -1997099, 4529534, 25766844, 607986, -13222, 9677543, -32294889, -6456008 } + }, + { + { -2444496, -149937, 29348902, 8186665, 1873760, 12489863, -30934579, -7839692, -7852844, -8138429 }, + { -15236356, -15433509, 7766470, 746860, 26346930, -10221762, -27333451, 10754588, -9431476, 5203576 }, + { 31834314, 14135496, -770007, 5159118, 20917671, -16768096, -7467973, -7337524, 31809243, 7347066 } + }, + { + { -9606723, -11874240, 20414459, 13033986, 13716524, -11691881, 19797970, -12211255, 15192876, -2087490 }, + { -12663563, -2181719, 1168162, -3804809, 26747877, -14138091, 10609330, 12694420, 33473243, -13382104 }, + { 33184999, 11180355, 15832085, -11385430, -1633671, 225884, 15089336, -11023903, -6135662, 14480053 } + }, + { + { 31308717, -5619998, 31030840, -1897099, 15674547, -6582883, 5496208, 13685227, 27595050, 8737275 }, + { -20318852, -15150239, 10933843, -16178022, 8335352, -7546022, -31008351, -12610604, 26498114, 66511 }, + { 22644454, -8761729, -16671776, 4884562, -3105614, -13559366, 30540766, -4286747, -13327787, -7515095 } + }, + { + { -28017847, 9834845, 18617207, -2681312, -3401956, -13307506, 8205540, 13585437, -17127465, 15115439 }, + { 23711543, -672915, 31206561, -8362711, 6164647, -9709987, -33535882, -1426096, 8236921, 16492939 }, + { -23910559, -13515526, -26299483, -4503841, 25005590, -7687270, 19574902, 10071562, 6708380, -6222424 } + }, + { + { 2101391, -4930054, 19702731, 2367575, -15427167, 1047675, 5301017, 9328700, 29955601, -11678310 }, + { 3096359, 9271816, -21620864, -15521844, -14847996, -7592937, -25892142, -12635595, -9917575, 6216608 }, + { -32615849, 338663, -25195611, 2510422, -29213566, -13820213, 24822830, -6146567, -26767480, 7525079 } + }, + { + { -23066649, -13985623, 16133487, -7896178, -3389565, 778788, -910336, -2782495, -19386633, 11994101 }, + { 21691500, -13624626, -641331, -14367021, 3285881, -3483596, -25064666, 9718258, -7477437, 13381418 }, + { 18445390, -4202236, 14979846, 11622458, -1727110, -3582980, 23111648, -6375247, 28535282, 15779576 } + }, + { + { 30098053, 3089662, -9234387, 16662135, -21306940, 11308411, -14068454, 12021730, 9955285, -16303356 }, + { 9734894, -14576830, -7473633, -9138735, 2060392, 11313496, -18426029, 9924399, 20194861, 13380996 }, + { -26378102, -7965207, -22167821, 15789297, -18055342, -6168792, -1984914, 15707771, 26342023, 10146099 } + } +}, +{ /* 30/31 */ + { + { -26016874, -219943, 21339191, -41388, 19745256, -2878700, -29637280, 2227040, 21612326, -545728 }, + { -13077387, 1184228, 23562814, -5970442, -20351244, -6348714, 25764461, 12243797, -20856566, 11649658 }, + { -10031494, 11262626, 27384172, 2271902, 26947504, -15997771, 39944, 6114064, 33514190, 2333242 } + }, + { + { -21433588, -12421821, 8119782, 7219913, -21830522, -9016134, -6679750, -12670638, 24350578, -13450001 }, + { -4116307, -11271533, -23886186, 4843615, -30088339, 690623, -31536088, -10406836, 8317860, 12352766 }, + { 18200138, -14475911, -33087759, -2696619, -23702521, -9102511, -23552096, -2287550, 20712163, 6719373 } + }, + { + { 26656208, 6075253, -7858556, 1886072, -28344043, 4262326, 11117530, -3763210, 26224235, -3297458 }, + { -17168938, -14854097, -3395676, -16369877, -19954045, 14050420, 21728352, 9493610, 18620611, -16428628 }, + { -13323321, 13325349, 11432106, 5964811, 18609221, 6062965, -5269471, -9725556, -30701573, -16479657 } + }, + { + { -23860538, -11233159, 26961357, 1640861, -32413112, -16737940, 12248509, -5240639, 13735342, 1934062 }, + { 25089769, 6742589, 17081145, -13406266, 21909293, -16067981, -15136294, -3765346, -21277997, 5473616 }, + { 31883677, -7961101, 1083432, -11572403, 22828471, 13290673, -7125085, 12469656, 29111212, -5451014 } + }, + { + { 24244947, -15050407, -26262976, 2791540, -14997599, 16666678, 24367466, 6388839, -10295587, 452383 }, + { -25640782, -3417841, 5217916, 16224624, 19987036, -4082269, -24236251, -5915248, 15766062, 8407814 }, + { -20406999, 13990231, 15495425, 16395525, 5377168, 15166495, -8917023, -4388953, -8067909, 2276718 } + }, + { + { 30157918, 12924066, -17712050, 9245753, 19895028, 3368142, -23827587, 5096219, 22740376, -7303417 }, + { 2041139, -14256350, 7783687, 13876377, -25946985, -13352459, 24051124, 13742383, -15637599, 13295222 }, + { 33338237, -8505733, 12532113, 7977527, 9106186, -1715251, -17720195, -4612972, -4451357, -14669444 } + }, + { + { -20045281, 5454097, -14346548, 6447146, 28862071, 1883651, -2469266, -4141880, 7770569, 9620597 }, + { 23208068, 7979712, 33071466, 8149229, 1758231, -10834995, 30945528, -1694323, -33502340, -14767970 }, + { 1439958, -16270480, -1079989, -793782, 4625402, 10647766, -5043801, 1220118, 30494170, -11440799 } + }, + { + { -5037580, -13028295, -2970559, -3061767, 15640974, -6701666, -26739026, 926050, -1684339, -13333647 }, + { 13908495, -3549272, 30919928, -6273825, -21521863, 7989039, 9021034, 9078865, 3353509, 4033511 }, + { -29663431, -15113610, 32259991, -344482, 24295849, -12912123, 23161163, 8839127, 27485041, 7356032 } + } +}, +{ /* 31/31 */ + { + { 9661027, 705443, 11980065, -5370154, -1628543, 14661173, -6346142, 2625015, 28431036, -16771834 }, + { -23839233, -8311415, -25945511, 7480958, -17681669, -8354183, -22545972, 14150565, 15970762, 4099461 }, + { 29262576, 16756590, 26350592, -8793563, 8529671, -11208050, 13617293, -9937143, 11465739, 8317062 } + }, + { + { -25493081, -6962928, 32500200, -9419051, -23038724, -2302222, 14898637, 3848455, 20969334, -5157516 }, + { -20384450, -14347713, -18336405, 13884722, -33039454, 2842114, -21610826, -3649888, 11177095, 14989547 }, + { -24496721, -11716016, 16959896, 2278463, 12066309, 10137771, 13515641, 2581286, -28487508, 9930240 } + }, + { + { -17751622, -2097826, 16544300, -13009300, -15914807, -14949081, 18345767, -13403753, 16291481, -5314038 }, + { -33229194, 2553288, 32678213, 9875984, 8534129, 6889387, -9676774, 6957617, 4368891, 9788741 }, + { 16660756, 7281060, -10830758, 12911820, 20108584, -8101676, -21722536, -8613148, 16250552, -11111103 } + }, + { + { -19765507, 2390526, -16551031, 14161980, 1905286, 6414907, 4689584, 10604807, -30190403, 4782747 }, + { -1354539, 14736941, -7367442, -13292886, 7710542, -14155590, -9981571, 4383045, 22546403, 437323 }, + { 31665577, -12180464, -16186830, 1491339, -18368625, 3294682, 27343084, 2786261, -30633590, -14097016 } + }, + { + { -14467279, -683715, -33374107, 7448552, 19294360, 14334329, -19690631, 2355319, -19284671, -6114373 }, + { 15121312, -15796162, 6377020, -6031361, -10798111, -12957845, 18952177, 15496498, -29380133, 11754228 }, + { -2637277, -13483075, 8488727, -14303896, 12728761, -1622493, 7141596, 11724556, 22761615, -10134141 } + }, + { + { 16918416, 11729663, -18083579, 3022987, -31015732, -13339659, -28741185, -12227393, 32851222, 11717399 }, + { 11166634, 7338049, -6722523, 4531520, -29468672, -7302055, 31474879, 3483633, -1193175, -4030831 }, + { -185635, 9921305, 31456609, -13536438, -12013818, 13348923, 33142652, 6546660, -19985279, -3948376 } + }, + { + { -32460596, 11266712, -11197107, -7899103, 31703694, 3855903, -8537131, -12833048, -30772034, -15486313 }, + { -18006477, 12709068, 3991746, -6479188, -21491523, -10550425, -31135347, -16049879, 10928917, 3011958 }, + { -6957757, -15594337, 31696059, 334240, 29576716, 14796075, -30831056, -12805180, 18008031, 10258577 } + }, + { + { -22448644, 15655569, 7018479, -4410003, -30314266, -1201591, -1853465, 1367120, 25127874, 6671743 }, + { 29701166, -14373934, -10878120, 9279288, -17568, 13127210, 21382910, 11042292, 25838796, 4642684 }, + { -20430234, 14955537, -24126347, 8124619, -5369288, -5990470, 30468147, -13900640, 18423289, 4177476 } + } +} diff --git a/sodium/sodium_fe_25_5_base2.h b/sodium/sodium_fe_25_5_base2.h new file mode 100644 index 00000000..90a1457e --- /dev/null +++ b/sodium/sodium_fe_25_5_base2.h @@ -0,0 +1,40 @@ +{ + { 25967493, -14356035, 29566456, 3660896, -12694345, 4014787, 27544626, -11754271, -6079156, 2047605 }, + { -12545711, 934262, -2722910, 3049990, -727428, 9406986, 12720692, 5043384, 19500929, -15469378 }, + { -8738181, 4489570, 9688441, -14785194, 10184609, -12363380, 29287919, 11864899, -24514362, -4438546 } +}, +{ + { 15636291, -9688557, 24204773, -7912398, 616977, -16685262, 27787600, -14772189, 28944400, -1550024 }, + { 16568933, 4717097, -11556148, -1102322, 15682896, -11807043, 16354577, -11775962, 7689662, 11199574 }, + { 30464156, -5976125, -11779434, -15670865, 23220365, 15915852, 7512774, 10017326, -17749093, -9920357 } +}, +{ + { 10861363, 11473154, 27284546, 1981175, -30064349, 12577861, 32867885, 14515107, -15438304, 10819380 }, + { 4708026, 6336745, 20377586, 9066809, -11272109, 6594696, -25653668, 12483688, -12668491, 5581306 }, + { 19563160, 16186464, -29386857, 4097519, 10237984, -4348115, 28542350, 13850243, -23678021, -15815942 } +}, +{ + { 5153746, 9909285, 1723747, -2777874, 30523605, 5516873, 19480852, 5230134, -23952439, -15175766 }, + { -30269007, -3463509, 7665486, 10083793, 28475525, 1649722, 20654025, 16520125, 30598449, 7715701 }, + { 28881845, 14381568, 9657904, 3680757, -20181635, 7843316, -31400660, 1370708, 29794553, -1409300 } +}, +{ + { -22518993, -6692182, 14201702, -8745502, -23510406, 8844726, 18474211, -1361450, -13062696, 13821877 }, + { -6455177, -7839871, 3374702, -4740862, -27098617, -10571707, 31655028, -7212327, 18853322, -14220951 }, + { 4566830, -12963868, -28974889, -12240689, -7602672, -2830569, -8514358, -10431137, 2207753, -3209784 } +}, +{ + { -25154831, -4185821, 29681144, 7868801, -6854661, -9423865, -12437364, -663000, -31111463, -16132436 }, + { 25576264, -2703214, 7349804, -11814844, 16472782, 9300885, 3844789, 15725684, 171356, 6466918 }, + { 23103977, 13316479, 9739013, -16149481, 817875, -15038942, 8965339, -14088058, -30714912, 16193877 } +}, +{ + { -33521811, 3180713, -2394130, 14003687, -16903474, -16270840, 17238398, 4729455, -18074513, 9256800 }, + { -25182317, -4174131, 32336398, 5036987, -21236817, 11360617, 22616405, 9761698, -19827198, 630305 }, + { -13720693, 2639453, -24237460, -7406481, 9494427, -5774029, -6554551, -15960994, -2449256, -14291300 } +}, +{ + { -3151181, -5046075, 9282714, 6866145, -31907062, -863023, -18940575, 15033784, 25105118, -7894876 }, + { -24326370, 15950226, -31801215, -14592823, -11662737, -5090925, 1573892, -2625887, 2198790, -15804619 }, + { -3099351, 10324967, -2241613, 7453183, -5446979, -2735503, -13812022, -16236442, -32461234, -12290683 } +} diff --git a/sodium/sodium_fe_25_5_constants.h b/sodium/sodium_fe_25_5_constants.h new file mode 100644 index 00000000..3dc9156a --- /dev/null +++ b/sodium/sodium_fe_25_5_constants.h @@ -0,0 +1,20 @@ +/* 37095705934669439343138083508754565189542113879843219016388785533085940283555 */ +static const fe25519 d = { + -10913610, 13857413, -15372611, 6949391, 114729, -8787816, -6275908, -3247719, -18696448, -12055116 +}; + +/* 2 * d = + * 16295367250680780974490674513165176452449235426866156013048779062215315747161 + */ +static const fe25519 d2 = { + -21827239, -5839606, -30745221, 13898782, 229458, 15978800, -12551817, -6495438, 29715968, 9444199 }; + +/* sqrt(-1) */ +static const fe25519 sqrtm1 = { + -32595792, -7943725, 9377950, 3500415, 12389472, -272473, -25146209, -2005654, 326686, 11406482 +}; + +/* A = 486662 */ +static const fe25519 curve25519_A = { + 486662, 0, 0, 0, 0, 0, 0, 0, 0, 0 +}; diff --git a/sodium/sodium_fe_25_5_fe.h b/sodium/sodium_fe_25_5_fe.h new file mode 100644 index 00000000..f216669e --- /dev/null +++ b/sodium/sodium_fe_25_5_fe.h @@ -0,0 +1,220 @@ +/* + Ignores top bit of h. + */ + +void +fe25519_frombytes(fe25519 h, const unsigned char *s) +{ + int64_t h0 = load_4(s); + int64_t h1 = load_3(s + 4) << 6; + int64_t h2 = load_3(s + 7) << 5; + int64_t h3 = load_3(s + 10) << 3; + int64_t h4 = load_3(s + 13) << 2; + int64_t h5 = load_4(s + 16); + int64_t h6 = load_3(s + 20) << 7; + int64_t h7 = load_3(s + 23) << 5; + int64_t h8 = load_3(s + 26) << 4; + int64_t h9 = (load_3(s + 29) & 8388607) << 2; + + int64_t carry0; + int64_t carry1; + int64_t carry2; + int64_t carry3; + int64_t carry4; + int64_t carry5; + int64_t carry6; + int64_t carry7; + int64_t carry8; + int64_t carry9; + + carry9 = (h9 + (int64_t)(1L << 24)) >> 25; + h0 += carry9 * 19; + h9 -= carry9 * ((uint64_t) 1L << 25); + carry1 = (h1 + (int64_t)(1L << 24)) >> 25; + h2 += carry1; + h1 -= carry1 * ((uint64_t) 1L << 25); + carry3 = (h3 + (int64_t)(1L << 24)) >> 25; + h4 += carry3; + h3 -= carry3 * ((uint64_t) 1L << 25); + carry5 = (h5 + (int64_t)(1L << 24)) >> 25; + h6 += carry5; + h5 -= carry5 * ((uint64_t) 1L << 25); + carry7 = (h7 + (int64_t)(1L << 24)) >> 25; + h8 += carry7; + h7 -= carry7 * ((uint64_t) 1L << 25); + + carry0 = (h0 + (int64_t)(1L << 25)) >> 26; + h1 += carry0; + h0 -= carry0 * ((uint64_t) 1L << 26); + carry2 = (h2 + (int64_t)(1L << 25)) >> 26; + h3 += carry2; + h2 -= carry2 * ((uint64_t) 1L << 26); + carry4 = (h4 + (int64_t)(1L << 25)) >> 26; + h5 += carry4; + h4 -= carry4 * ((uint64_t) 1L << 26); + carry6 = (h6 + (int64_t)(1L << 25)) >> 26; + h7 += carry6; + h6 -= carry6 * ((uint64_t) 1L << 26); + carry8 = (h8 + (int64_t)(1L << 25)) >> 26; + h9 += carry8; + h8 -= carry8 * ((uint64_t) 1L << 26); + + h[0] = (int32_t) h0; + h[1] = (int32_t) h1; + h[2] = (int32_t) h2; + h[3] = (int32_t) h3; + h[4] = (int32_t) h4; + h[5] = (int32_t) h5; + h[6] = (int32_t) h6; + h[7] = (int32_t) h7; + h[8] = (int32_t) h8; + h[9] = (int32_t) h9; +} + +/* + Preconditions: + |h| bounded by 1.1*2^26,1.1*2^25,1.1*2^26,1.1*2^25,etc. + + Write p=2^255-19; q=floor(h/p). + Basic claim: q = floor(2^(-255)(h + 19 2^(-25)h9 + 2^(-1))). + + Proof: + Have |h|<=p so |q|<=1 so |19^2 2^(-255) q|<1/4. + Also have |h-2^230 h9|<2^231 so |19 2^(-255)(h-2^230 h9)|<1/4. + + Write y=2^(-1)-19^2 2^(-255)q-19 2^(-255)(h-2^230 h9). + Then 0> 25; + q = (h0 + q) >> 26; + q = (h1 + q) >> 25; + q = (h2 + q) >> 26; + q = (h3 + q) >> 25; + q = (h4 + q) >> 26; + q = (h5 + q) >> 25; + q = (h6 + q) >> 26; + q = (h7 + q) >> 25; + q = (h8 + q) >> 26; + q = (h9 + q) >> 25; + + /* Goal: Output h-(2^255-19)q, which is between 0 and 2^255-20. */ + h0 += 19 * q; + /* Goal: Output h-2^255 q, which is between 0 and 2^255-20. */ + + carry0 = h0 >> 26; + h1 += carry0; + h0 -= carry0 * ((uint32_t) 1L << 26); + carry1 = h1 >> 25; + h2 += carry1; + h1 -= carry1 * ((uint32_t) 1L << 25); + carry2 = h2 >> 26; + h3 += carry2; + h2 -= carry2 * ((uint32_t) 1L << 26); + carry3 = h3 >> 25; + h4 += carry3; + h3 -= carry3 * ((uint32_t) 1L << 25); + carry4 = h4 >> 26; + h5 += carry4; + h4 -= carry4 * ((uint32_t) 1L << 26); + carry5 = h5 >> 25; + h6 += carry5; + h5 -= carry5 * ((uint32_t) 1L << 25); + carry6 = h6 >> 26; + h7 += carry6; + h6 -= carry6 * ((uint32_t) 1L << 26); + carry7 = h7 >> 25; + h8 += carry7; + h7 -= carry7 * ((uint32_t) 1L << 25); + carry8 = h8 >> 26; + h9 += carry8; + h8 -= carry8 * ((uint32_t) 1L << 26); + carry9 = h9 >> 25; + h9 -= carry9 * ((uint32_t) 1L << 25); + + h[0] = h0; + h[1] = h1; + h[2] = h2; + h[3] = h3; + h[4] = h4; + h[5] = h5; + h[6] = h6; + h[7] = h7; + h[8] = h8; + h[9] = h9; +} + +/* + Goal: Output h0+...+2^255 h10-2^255 q, which is between 0 and 2^255-20. + Have h0+...+2^230 h9 between 0 and 2^255-1; + evidently 2^255 h10-2^255 q = 0. + + Goal: Output h0+...+2^230 h9. + */ + +void +fe25519_tobytes(unsigned char *s, const fe25519 h) +{ + fe25519 t; + + fe25519_reduce(t, h); + s[0] = t[0] >> 0; + s[1] = t[0] >> 8; + s[2] = t[0] >> 16; + s[3] = (t[0] >> 24) | (t[1] * ((uint32_t) 1 << 2)); + s[4] = t[1] >> 6; + s[5] = t[1] >> 14; + s[6] = (t[1] >> 22) | (t[2] * ((uint32_t) 1 << 3)); + s[7] = t[2] >> 5; + s[8] = t[2] >> 13; + s[9] = (t[2] >> 21) | (t[3] * ((uint32_t) 1 << 5)); + s[10] = t[3] >> 3; + s[11] = t[3] >> 11; + s[12] = (t[3] >> 19) | (t[4] * ((uint32_t) 1 << 6)); + s[13] = t[4] >> 2; + s[14] = t[4] >> 10; + s[15] = t[4] >> 18; + s[16] = t[5] >> 0; + s[17] = t[5] >> 8; + s[18] = t[5] >> 16; + s[19] = (t[5] >> 24) | (t[6] * ((uint32_t) 1 << 1)); + s[20] = t[6] >> 7; + s[21] = t[6] >> 15; + s[22] = (t[6] >> 23) | (t[7] * ((uint32_t) 1 << 3)); + s[23] = t[7] >> 5; + s[24] = t[7] >> 13; + s[25] = (t[7] >> 21) | (t[8] * ((uint32_t) 1 << 4)); + s[26] = t[8] >> 4; + s[27] = t[8] >> 12; + s[28] = (t[8] >> 20) | (t[9] * ((uint32_t) 1 << 6)); + s[29] = t[9] >> 2; + s[30] = t[9] >> 10; + s[31] = t[9] >> 18; +} diff --git a/sodium/sodium_fe_51_base.h b/sodium/sodium_fe_51_base.h new file mode 100644 index 00000000..6b3b833e --- /dev/null +++ b/sodium/sodium_fe_51_base.h @@ -0,0 +1,1344 @@ +{ /* 0/31 */ + { + { 1288382639258501, 245678601348599, 269427782077623, 1462984067271730, 137412439391563 }, + { 62697248952638, 204681361388450, 631292143396476, 338455783676468, 1213667448819585 }, + { 301289933810280, 1259582250014073, 1422107436869536, 796239922652654, 1953934009299142 } + }, + { + { 1380971894829527, 790832306631236, 2067202295274102, 1995808275510000, 1566530869037010 }, + { 463307831301544, 432984605774163, 1610641361907204, 750899048855000, 1894842303421586 }, + { 748439484463711, 1033211726465151, 1396005112841647, 1611506220286469, 1972177495910992 } + }, + { + { 1601611775252272, 1720807796594148, 1132070835939856, 1260455018889551, 2147779492816911 }, + { 316559037616741, 2177824224946892, 1459442586438991, 1461528397712656, 751590696113597 }, + { 1850748884277385, 1200145853858453, 1068094770532492, 672251375690438, 1586055907191707 } + }, + { + { 934282339813791, 1846903124198670, 1172395437954843, 1007037127761661, 1830588347719256 }, + { 1694390458783935, 1735906047636159, 705069562067493, 648033061693059, 696214010414170 }, + { 1121406372216585, 192876649532226, 190294192191717, 1994165897297032, 2245000007398739 } + }, + { + { 769950342298419, 132954430919746, 844085933195555, 974092374476333, 726076285546016 }, + { 425251763115706, 608463272472562, 442562545713235, 837766094556764, 374555092627893 }, + { 1086255230780037, 274979815921559, 1960002765731872, 929474102396301, 1190409889297339 } + }, + { + { 1388594989461809, 316767091099457, 394298842192982, 1230079486801005, 1440737038838979 }, + { 7380825640100, 146210432690483, 304903576448906, 1198869323871120, 997689833219095 }, + { 1181317918772081, 114573476638901, 262805072233344, 265712217171332, 294181933805782 } + }, + { + { 665000864555967, 2065379846933859, 370231110385876, 350988370788628, 1233371373142985 }, + { 2019367628972465, 676711900706637, 110710997811333, 1108646842542025, 517791959672113 }, + { 965130719900578, 247011430587952, 526356006571389, 91986625355052, 2157223321444601 } + }, + { + { 2068619540119183, 1966274918058806, 957728544705549, 729906502578991, 159834893065166 }, + { 2073601412052185, 31021124762708, 264500969797082, 248034690651703, 1030252227928288 }, + { 551790716293402, 1989538725166328, 801169423371717, 2052451893578887, 678432056995012 } + } +}, +{ /* 1/31 */ + { + { 1368953770187805, 790347636712921, 437508475667162, 2142576377050580, 1932081720066286 }, + { 953638594433374, 1092333936795051, 1419774766716690, 805677984380077, 859228993502513 }, + { 1200766035879111, 20142053207432, 1465634435977050, 1645256912097844, 295121984874596 } + }, + { + { 1735718747031557, 1248237894295956, 1204753118328107, 976066523550493, 65943769534592 }, + { 1060098822528990, 1586825862073490, 212301317240126, 1975302711403555, 666724059764335 }, + { 1091990273418756, 1572899409348578, 80968014455247, 306009358661350, 1520450739132526 } + }, + { + { 1480517209436112, 1511153322193952, 1244343858991172, 304788150493241, 369136856496443 }, + { 2151330273626164, 762045184746182, 1688074332551515, 823046109005759, 907602769079491 }, + { 2047386910586836, 168470092900250, 1552838872594810, 340951180073789, 360819374702533 } + }, + { + { 1982622644432056, 2014393600336956, 128909208804214, 1617792623929191, 105294281913815 }, + { 980234343912898, 1712256739246056, 588935272190264, 204298813091998, 841798321043288 }, + { 197561292938973, 454817274782871, 1963754960082318, 2113372252160468, 971377527342673 } + }, + { + { 164699448829328, 3127451757672, 1199504971548753, 1766155447043652, 1899238924683527 }, + { 732262946680281, 1674412764227063, 2182456405662809, 1350894754474250, 558458873295247 }, + { 2103305098582922, 1960809151316468, 715134605001343, 1454892949167181, 40827143824949 } + }, + { + { 1239289043050212, 1744654158124578, 758702410031698, 1796762995074688, 1603056663766 }, + { 2232056027107988, 987343914584615, 2115594492994461, 1819598072792159, 1119305654014850 }, + { 320153677847348, 939613871605645, 641883205761567, 1930009789398224, 329165806634126 } + }, + { + { 980930490474130, 1242488692177893, 1251446316964684, 1086618677993530, 1961430968465772 }, + { 276821765317453, 1536835591188030, 1305212741412361, 61473904210175, 2051377036983058 }, + { 833449923882501, 1750270368490475, 1123347002068295, 185477424765687, 278090826653186 } + }, + { + { 794524995833413, 1849907304548286, 53348672473145, 1272368559505217, 1147304168324779 }, + { 1504846112759364, 1203096289004681, 562139421471418, 274333017451844, 1284344053775441 }, + { 483048732424432, 2116063063343382, 30120189902313, 292451576741007, 1156379271702225 } + } +}, +{ /* 2/31 */ + { + { 928372153029038, 2147692869914564, 1455665844462196, 1986737809425946, 185207050258089 }, + { 137732961814206, 706670923917341, 1387038086865771, 1965643813686352, 1384777115696347 }, + { 481144981981577, 2053319313589856, 2065402289827512, 617954271490316, 1106602634668125 } + }, + { + { 696298019648792, 893299659040895, 1148636718636009, 26734077349617, 2203955659340681 }, + { 657390353372855, 998499966885562, 991893336905797, 810470207106761, 343139804608786 }, + { 791736669492960, 934767652997115, 824656780392914, 1759463253018643, 361530362383518 } + }, + { + { 2022541353055597, 2094700262587466, 1551008075025686, 242785517418164, 695985404963562 }, + { 1287487199965223, 2215311941380308, 1552928390931986, 1664859529680196, 1125004975265243 }, + { 677434665154918, 989582503122485, 1817429540898386, 1052904935475344, 1143826298169798 } + }, + { + { 367266328308408, 318431188922404, 695629353755355, 634085657580832, 24581612564426 }, + { 773360688841258, 1815381330538070, 363773437667376, 539629987070205, 783280434248437 }, + { 180820816194166, 168937968377394, 748416242794470, 1227281252254508, 1567587861004268 } + }, + { + { 478775558583645, 2062896624554807, 699391259285399, 358099408427873, 1277310261461761 }, + { 1984740906540026, 1079164179400229, 1056021349262661, 1659958556483663, 1088529069025527 }, + { 580736401511151, 1842931091388998, 1177201471228238, 2075460256527244, 1301133425678027 } + }, + { + { 1515728832059182, 1575261009617579, 1510246567196186, 191078022609704, 116661716289141 }, + { 1295295738269652, 1714742313707026, 545583042462581, 2034411676262552, 1513248090013606 }, + { 230710545179830, 30821514358353, 760704303452229, 390668103790604, 573437871383156 } + }, + { + { 1169380107545646, 263167233745614, 2022901299054448, 819900753251120, 2023898464874585 }, + { 2102254323485823, 1570832666216754, 34696906544624, 1993213739807337, 70638552271463 }, + { 894132856735058, 548675863558441, 845349339503395, 1942269668326667, 1615682209874691 } + }, + { + { 1287670217537834, 1222355136884920, 1846481788678694, 1150426571265110, 1613523400722047 }, + { 793388516527298, 1315457083650035, 1972286999342417, 1901825953052455, 338269477222410 }, + { 550201530671806, 778605267108140, 2063911101902983, 115500557286349, 2041641272971022 } + } +}, +{ /* 3/31 */ + { + { 717255318455100, 519313764361315, 2080406977303708, 541981206705521, 774328150311600 }, + { 261715221532238, 1795354330069993, 1496878026850283, 499739720521052, 389031152673770 }, + { 1997217696294013, 1717306351628065, 1684313917746180, 1644426076011410, 1857378133465451 } + }, + { + { 1475434724792648, 76931896285979, 1116729029771667, 2002544139318042, 725547833803938 }, + { 2022306639183567, 726296063571875, 315345054448644, 1058733329149221, 1448201136060677 }, + { 1710065158525665, 1895094923036397, 123988286168546, 1145519900776355, 1607510767693874 } + }, + { + { 561605375422540, 1071733543815037, 131496498800990, 1946868434569999, 828138133964203 }, + { 1548495173745801, 442310529226540, 998072547000384, 553054358385281, 644824326376171 }, + { 1445526537029440, 2225519789662536, 914628859347385, 1064754194555068, 1660295614401091 } + }, + { + { 1199690223111956, 24028135822341, 66638289244341, 57626156285975, 565093967979607 }, + { 876926774220824, 554618976488214, 1012056309841565, 839961821554611, 1414499340307677 }, + { 703047626104145, 1266841406201770, 165556500219173, 486991595001879, 1011325891650656 } + }, + { + { 1622861044480487, 1156394801573634, 1869132565415504, 327103985777730, 2095342781472284 }, + { 334886927423922, 489511099221528, 129160865966726, 1720809113143481, 619700195649254 }, + { 1646545795166119, 1758370782583567, 714746174550637, 1472693650165135, 898994790308209 } + }, + { + { 333403773039279, 295772542452938, 1693106465353610, 912330357530760, 471235657950362 }, + { 1811196219982022, 1068969825533602, 289602974833439, 1988956043611592, 863562343398367 }, + { 906282429780072, 2108672665779781, 432396390473936, 150625823801893, 1708930497638539 } + }, + { + { 925664675702328, 21416848568684, 1831436641861340, 601157008940113, 371818055044496 }, + { 1479786007267725, 1738881859066675, 68646196476567, 2146507056100328, 1247662817535471 }, + { 52035296774456, 939969390708103, 312023458773250, 59873523517659, 1231345905848899 } + }, + { + { 643355106415761, 290186807495774, 2013561737429023, 319648069511546, 393736678496162 }, + { 129358342392716, 1932811617704777, 1176749390799681, 398040349861790, 1170779668090425 }, + { 2051980782668029, 121859921510665, 2048329875753063, 1235229850149665, 519062146124755 } + } +}, +{ /* 4/31 */ + { + { 1608170971973096, 415809060360428, 1350468408164766, 2038620059057678, 1026904485989112 }, + { 1837656083115103, 1510134048812070, 906263674192061, 1821064197805734, 565375124676301 }, + { 578027192365650, 2034800251375322, 2128954087207123, 478816193810521, 2196171989962750 } + }, + { + { 1633188840273139, 852787172373708, 1548762607215796, 1266275218902681, 1107218203325133 }, + { 462189358480054, 1784816734159228, 1611334301651368, 1303938263943540, 707589560319424 }, + { 1038829280972848, 38176604650029, 753193246598573, 1136076426528122, 595709990562434 } + }, + { + { 1408451820859834, 2194984964010833, 2198361797561729, 1061962440055713, 1645147963442934 }, + { 4701053362120, 1647641066302348, 1047553002242085, 1923635013395977, 206970314902065 }, + { 1750479161778571, 1362553355169293, 1891721260220598, 966109370862782, 1024913988299801 } + }, + { + { 212699049131723, 1117950018299775, 1873945661751056, 1403802921984058, 130896082652698 }, + { 636808533673210, 1262201711667560, 390951380330599, 1663420692697294, 561951321757406 }, + { 520731594438141, 1446301499955692, 273753264629267, 1565101517999256, 1019411827004672 } + }, + { + { 926527492029409, 1191853477411379, 734233225181171, 184038887541270, 1790426146325343 }, + { 1464651961852572, 1483737295721717, 1519450561335517, 1161429831763785, 405914998179977 }, + { 996126634382301, 796204125879525, 127517800546509, 344155944689303, 615279846169038 } + }, + { + { 738724080975276, 2188666632415296, 1961313708559162, 1506545807547587, 1151301638969740 }, + { 622917337413835, 1218989177089035, 1284857712846592, 970502061709359, 351025208117090 }, + { 2067814584765580, 1677855129927492, 2086109782475197, 235286517313238, 1416314046739645 } + }, + { + { 586844262630358, 307444381952195, 458399356043426, 602068024507062, 1028548203415243 }, + { 678489922928203, 2016657584724032, 90977383049628, 1026831907234582, 615271492942522 }, + { 301225714012278, 1094837270268560, 1202288391010439, 644352775178361, 1647055902137983 } + }, + { + { 1210746697896478, 1416608304244708, 686487477217856, 1245131191434135, 1051238336855737 }, + { 1135604073198207, 1683322080485474, 769147804376683, 2086688130589414, 900445683120379 }, + { 1971518477615628, 401909519527336, 448627091057375, 1409486868273821, 1214789035034363 } + } +}, +{ /* 5/31 */ + { + { 1364039144731711, 1897497433586190, 2203097701135459, 145461396811251, 1349844460790699 }, + { 1045230323257973, 818206601145807, 630513189076103, 1672046528998132, 807204017562437 }, + { 439961968385997, 386362664488986, 1382706320807688, 309894000125359, 2207801346498567 } + }, + { + { 1229004686397588, 920643968530863, 123975893911178, 681423993215777, 1400559197080973 }, + { 2003766096898049, 170074059235165, 1141124258967971, 1485419893480973, 1573762821028725 }, + { 729905708611432, 1270323270673202, 123353058984288, 426460209632942, 2195574535456672 } + }, + { + { 1271140255321235, 2044363183174497, 52125387634689, 1445120246694705, 942541986339084 }, + { 1761608437466135, 583360847526804, 1586706389685493, 2157056599579261, 1170692369685772 }, + { 871476219910823, 1878769545097794, 2241832391238412, 548957640601001, 690047440233174 } + }, + { + { 297194732135507, 1366347803776820, 1301185512245601, 561849853336294, 1533554921345731 }, + { 999628998628371, 1132836708493400, 2084741674517453, 469343353015612, 678782988708035 }, + { 2189427607417022, 699801937082607, 412764402319267, 1478091893643349, 2244675696854460 } + }, + { + { 1712292055966563, 204413590624874, 1405738637332841, 408981300829763, 861082219276721 }, + { 508561155940631, 966928475686665, 2236717801150132, 424543858577297, 2089272956986143 }, + { 221245220129925, 1156020201681217, 491145634799213, 542422431960839, 828100817819207 } + }, + { + { 153756971240384, 1299874139923977, 393099165260502, 1058234455773022, 996989038681183 }, + { 559086812798481, 573177704212711, 1629737083816402, 1399819713462595, 1646954378266038 }, + { 1887963056288059, 228507035730124, 1468368348640282, 930557653420194, 613513962454686 } + }, + { + { 1224529808187553, 1577022856702685, 2206946542980843, 625883007765001, 279930793512158 }, + { 1076287717051609, 1114455570543035, 187297059715481, 250446884292121, 1885187512550540 }, + { 902497362940219, 76749815795675, 1657927525633846, 1420238379745202, 1340321636548352 } + }, + { + { 1129576631190784, 1281994010027327, 996844254743018, 257876363489249, 1150850742055018 }, + { 628740660038789, 1943038498527841, 467786347793886, 1093341428303375, 235413859513003 }, + { 237425418909360, 469614029179605, 1512389769174935, 1241726368345357, 441602891065214 } + } +}, +{ /* 6/31 */ + { + { 1736417953058555, 726531315520508, 1833335034432527, 1629442561574747, 624418919286085 }, + { 1960754663920689, 497040957888962, 1909832851283095, 1271432136996826, 2219780368020940 }, + { 1537037379417136, 1358865369268262, 2130838645654099, 828733687040705, 1999987652890901 } + }, + { + { 629042105241814, 1098854999137608, 887281544569320, 1423102019874777, 7911258951561 }, + { 1811562332665373, 1501882019007673, 2213763501088999, 359573079719636, 36370565049116 }, + { 218907117361280, 1209298913016966, 1944312619096112, 1130690631451061, 1342327389191701 } + }, + { + { 1369976867854704, 1396479602419169, 1765656654398856, 2203659200586299, 998327836117241 }, + { 2230701885562825, 1348173180338974, 2172856128624598, 1426538746123771, 444193481326151 }, + { 784210426627951, 918204562375674, 1284546780452985, 1324534636134684, 1872449409642708 } + }, + { + { 319638829540294, 596282656808406, 2037902696412608, 1557219121643918, 341938082688094 }, + { 1901860206695915, 2004489122065736, 1625847061568236, 973529743399879, 2075287685312905 }, + { 1371853944110545, 1042332820512553, 1949855697918254, 1791195775521505, 37487364849293 } + }, + { + { 687200189577855, 1082536651125675, 644224940871546, 340923196057951, 343581346747396 }, + { 2082717129583892, 27829425539422, 145655066671970, 1690527209845512, 1865260509673478 }, + { 1059729620568824, 2163709103470266, 1440302280256872, 1769143160546397, 869830310425069 } + }, + { + { 1609516219779025, 777277757338817, 2101121130363987, 550762194946473, 1905542338659364 }, + { 2024821921041576, 426948675450149, 595133284085473, 471860860885970, 600321679413000 }, + { 598474602406721, 1468128276358244, 1191923149557635, 1501376424093216, 1281662691293476 } + }, + { + { 1721138489890707, 1264336102277790, 433064545421287, 1359988423149466, 1561871293409447 }, + { 719520245587143, 393380711632345, 132350400863381, 1543271270810729, 1819543295798660 }, + { 396397949784152, 1811354474471839, 1362679985304303, 2117033964846756, 498041172552279 } + }, + { + { 1812471844975748, 1856491995543149, 126579494584102, 1036244859282620, 1975108050082550 }, + { 650623932407995, 1137551288410575, 2125223403615539, 1725658013221271, 2134892965117796 }, + { 522584000310195, 1241762481390450, 1743702789495384, 2227404127826575, 1686746002148897 } + } +}, +{ /* 7/31 */ + { + { 427904865186312, 1703211129693455, 1585368107547509, 1436984488744336, 761188534613978 }, + { 318101947455002, 248138407995851, 1481904195303927, 309278454311197, 1258516760217879 }, + { 1275068538599310, 513726919533379, 349926553492294, 688428871968420, 1702400196000666 } + }, + { + { 1061864036265233, 961611260325381, 321859632700838, 1045600629959517, 1985130202504038 }, + { 1558816436882417, 1962896332636523, 1337709822062152, 1501413830776938, 294436165831932 }, + { 818359826554971, 1862173000996177, 626821592884859, 573655738872376, 1749691246745455 } + }, + { + { 1988022651432119, 1082111498586040, 1834020786104821, 1454826876423687, 692929915223122 }, + { 2146513703733331, 584788900394667, 464965657279958, 2183973639356127, 238371159456790 }, + { 1129007025494441, 2197883144413266, 265142755578169, 971864464758890, 1983715884903702 } + }, + { + { 1291366624493075, 381456718189114, 1711482489312444, 1815233647702022, 892279782992467 }, + { 444548969917454, 1452286453853356, 2113731441506810, 645188273895859, 810317625309512 }, + { 2242724082797924, 1373354730327868, 1006520110883049, 2147330369940688, 1151816104883620 } + }, + { + { 1745720200383796, 1911723143175317, 2056329390702074, 355227174309849, 879232794371100 }, + { 163723479936298, 115424889803150, 1156016391581227, 1894942220753364, 1970549419986329 }, + { 681981452362484, 267208874112496, 1374683991933094, 638600984916117, 646178654558546 } + }, + { + { 13378654854251, 106237307029567, 1944412051589651, 1841976767925457, 230702819835573 }, + { 260683893467075, 854060306077237, 913639551980112, 4704576840123, 280254810808712 }, + { 715374893080287, 1173334812210491, 1806524662079626, 1894596008000979, 398905715033393 } + }, + { + { 500026409727661, 1596431288195371, 1420380351989370, 985211561521489, 392444930785633 }, + { 2096421546958141, 1922523000950363, 789831022876840, 427295144688779, 320923973161730 }, + { 1927770723575450, 1485792977512719, 1850996108474547, 551696031508956, 2126047405475647 } + }, + { + { 2112099158080148, 742570803909715, 6484558077432, 1951119898618916, 93090382703416 }, + { 383905201636970, 859946997631870, 855623867637644, 1017125780577795, 794250831877809 }, + { 77571826285752, 999304298101753, 487841111777762, 1038031143212339, 339066367948762 } + } +}, +{ /* 8/31 */ + { + { 674994775520533, 266035846330789, 826951213393478, 1405007746162285, 1781791018620876 }, + { 1001412661522686, 348196197067298, 1666614366723946, 888424995032760, 580747687801357 }, + { 1939560076207777, 1409892634407635, 552574736069277, 383854338280405, 190706709864139 } + }, + { + { 2177087163428741, 1439255351721944, 1208070840382793, 2230616362004769, 1396886392021913 }, + { 676962063230039, 1880275537148808, 2046721011602706, 888463247083003, 1318301552024067 }, + { 1466980508178206, 617045217998949, 652303580573628, 757303753529064, 207583137376902 } + }, + { + { 1511056752906902, 105403126891277, 493434892772846, 1091943425335976, 1802717338077427 }, + { 1853982405405128, 1878664056251147, 1528011020803992, 1019626468153565, 1128438412189035 }, + { 1963939888391106, 293456433791664, 697897559513649, 985882796904380, 796244541237972 } + }, + { + { 416770998629779, 389655552427054, 1314476859406756, 1749382513022778, 1161905598739491 }, + { 1428358296490651, 1027115282420478, 304840698058337, 441410174026628, 1819358356278573 }, + { 204943430200135, 1554861433819175, 216426658514651, 264149070665950, 2047097371738319 } + }, + { + { 1934415182909034, 1393285083565062, 516409331772960, 1157690734993892, 121039666594268 }, + { 662035583584445, 286736105093098, 1131773000510616, 818494214211439, 472943792054479 }, + { 665784778135882, 1893179629898606, 808313193813106, 276797254706413, 1563426179676396 } + }, + { + { 945205108984232, 526277562959295, 1324180513733566, 1666970227868664, 153547609289173 }, + { 2031433403516252, 203996615228162, 170487168837083, 981513604791390, 843573964916831 }, + { 1476570093962618, 838514669399805, 1857930577281364, 2017007352225784, 317085545220047 } + }, + { + { 1461557121912842, 1600674043318359, 2157134900399597, 1670641601940616, 127765583803283 }, + { 1293543509393474, 2143624609202546, 1058361566797508, 214097127393994, 946888515472729 }, + { 357067959932916, 1290876214345711, 521245575443703, 1494975468601005, 800942377643885 } + }, + { + { 566116659100033, 820247422481740, 994464017954148, 327157611686365, 92591318111744 }, + { 617256647603209, 1652107761099439, 1857213046645471, 1085597175214970, 817432759830522 }, + { 771808161440705, 1323510426395069, 680497615846440, 851580615547985, 1320806384849017 } + } +}, +{ /* 9/31 */ + { + { 1219260086131915, 647169006596815, 79601124759706, 2161724213426748, 404861897060198 }, + { 1327968293887866, 1335500852943256, 1401587164534264, 558137311952440, 1551360549268902 }, + { 417621685193956, 1429953819744454, 396157358457099, 1940470778873255, 214000046234152 } + }, + { + { 1268047918491973, 2172375426948536, 1533916099229249, 1761293575457130, 1590622667026765 }, + { 1627072914981959, 2211603081280073, 1912369601616504, 1191770436221309, 2187309757525860 }, + { 1149147819689533, 378692712667677, 828475842424202, 2218619146419342, 70688125792186 } + }, + { + { 1299739417079761, 1438616663452759, 1536729078504412, 2053896748919838, 1008421032591246 }, + { 2040723824657366, 399555637875075, 632543375452995, 872649937008051, 1235394727030233 }, + { 2211311599327900, 2139787259888175, 938706616835350, 12609661139114, 2081897930719789 } + }, + { + { 1324994503390450, 336982330582631, 1183998925654177, 1091654665913274, 48727673971319 }, + { 1845522914617879, 1222198248335542, 150841072760134, 1927029069940982, 1189913404498011 }, + { 1079559557592645, 2215338383666441, 1903569501302605, 49033973033940, 305703433934152 } + }, + { + { 94653405416909, 1386121349852999, 1062130477891762, 36553947479274, 833669648948846 }, + { 1432015813136298, 440364795295369, 1395647062821501, 1976874522764578, 934452372723352 }, + { 1296625309219774, 2068273464883862, 1858621048097805, 1492281814208508, 2235868981918946 } + }, + { + { 1490330266465570, 1858795661361448, 1436241134969763, 294573218899647, 1208140011028933 }, + { 1282462923712748, 741885683986255, 2027754642827561, 518989529541027, 1826610009555945 }, + { 1525827120027511, 723686461809551, 1597702369236987, 244802101764964, 1502833890372311 } + }, + { + { 113622036244513, 1233740067745854, 674109952278496, 2114345180342965, 166764512856263 }, + { 2041668749310338, 2184405322203901, 1633400637611036, 2110682505536899, 2048144390084644 }, + { 503058759232932, 760293024620937, 2027152777219493, 666858468148475, 1539184379870952 } + }, + { + { 1916168475367211, 915626432541343, 883217071712575, 363427871374304, 1976029821251593 }, + { 678039535434506, 570587290189340, 1605302676614120, 2147762562875701, 1706063797091704 }, + { 1439489648586438, 2194580753290951, 832380563557396, 561521973970522, 584497280718389 } + } +}, +{ /* 10/31 */ + { + { 187989455492609, 681223515948275, 1933493571072456, 1872921007304880, 488162364135671 }, + { 1413466089534451, 410844090765630, 1397263346404072, 408227143123410, 1594561803147811 }, + { 2102170800973153, 719462588665004, 1479649438510153, 1097529543970028, 1302363283777685 } + }, + { + { 942065717847195, 1069313679352961, 2007341951411051, 70973416446291, 1419433790163706 }, + { 1146565545556377, 1661971299445212, 406681704748893, 564452436406089, 1109109865829139 }, + { 2214421081775077, 1165671861210569, 1890453018796184, 3556249878661, 442116172656317 } + }, + { + { 753830546620811, 1666955059895019, 1530775289309243, 1119987029104146, 2164156153857580 }, + { 615171919212796, 1523849404854568, 854560460547503, 2067097370290715, 1765325848586042 }, + { 1094538949313667, 1796592198908825, 870221004284388, 2025558921863561, 1699010892802384 } + }, + { + { 1951351290725195, 1916457206844795, 198025184438026, 1909076887557595, 1938542290318919 }, + { 1014323197538413, 869150639940606, 1756009942696599, 1334952557375672, 1544945379082874 }, + { 764055910920305, 1603590757375439, 146805246592357, 1843313433854297, 954279890114939 } + }, + { + { 80113526615750, 764536758732259, 1055139345100233, 469252651759390, 617897512431515 }, + { 74497112547268, 740094153192149, 1745254631717581, 727713886503130, 1283034364416928 }, + { 525892105991110, 1723776830270342, 1476444848991936, 573789489857760, 133864092632978 } + }, + { + { 542611720192581, 1986812262899321, 1162535242465837, 481498966143464, 544600533583622 }, + { 64123227344372, 1239927720647794, 1360722983445904, 222610813654661, 62429487187991 }, + { 1793193323953132, 91096687857833, 70945970938921, 2158587638946380, 1537042406482111 } + }, + { + { 1895854577604609, 1394895708949416, 1728548428495944, 1140864900240149, 563645333603061 }, + { 141358280486863, 91435889572504, 1087208572552643, 1829599652522921, 1193307020643647 }, + { 1611230858525381, 950720175540785, 499589887488610, 2001656988495019, 88977313255908 } + }, + { + { 1189080501479658, 2184348804772597, 1040818725742319, 2018318290311834, 1712060030915354 }, + { 873966876953756, 1090638350350440, 1708559325189137, 672344594801910, 1320437969700239 }, + { 1508590048271766, 1131769479776094, 101550868699323, 428297785557897, 561791648661744 } + } +}, +{ /* 11/31 */ + { + { 756417570499462, 237882279232602, 2136263418594016, 1701968045454886, 703713185137472 }, + { 1781187809325462, 1697624151492346, 1381393690939988, 175194132284669, 1483054666415238 }, + { 2175517777364616, 708781536456029, 955668231122942, 1967557500069555, 2021208005604118 } + }, + { + { 1115135966606887, 224217372950782, 915967306279222, 593866251291540, 561747094208006 }, + { 1443163092879439, 391875531646162, 2180847134654632, 464538543018753, 1594098196837178 }, + { 850858855888869, 319436476624586, 327807784938441, 740785849558761, 17128415486016 } + }, + { + { 2132756334090067, 536247820155645, 48907151276867, 608473197600695, 1261689545022784 }, + { 1525176236978354, 974205476721062, 293436255662638, 148269621098039, 137961998433963 }, + { 1121075518299410, 2071745529082111, 1265567917414828, 1648196578317805, 496232102750820 } + }, + { + { 122321229299801, 1022922077493685, 2001275453369484, 2017441881607947, 993205880778002 }, + { 654925550560074, 1168810995576858, 575655959430926, 905758704861388, 496774564663534 }, + { 1954109525779738, 2117022646152485, 338102630417180, 1194140505732026, 107881734943492 } + }, + { + { 1714785840001267, 2036500018681589, 1876380234251966, 2056717182974196, 1645855254384642 }, + { 106431476499341, 62482972120563, 1513446655109411, 807258751769522, 538491469114 }, + { 2002850762893643, 1243624520538135, 1486040410574605, 2184752338181213, 378495998083531 } + }, + { + { 922510868424903, 1089502620807680, 402544072617374, 1131446598479839, 1290278588136533 }, + { 1867998812076769, 715425053580701, 39968586461416, 2173068014586163, 653822651801304 }, + { 162892278589453, 182585796682149, 75093073137630, 497037941226502, 133871727117371 } + }, + { + { 1914596576579670, 1608999621851578, 1987629837704609, 1519655314857977, 1819193753409464 }, + { 1949315551096831, 1069003344994464, 1939165033499916, 1548227205730856, 1933767655861407 }, + { 1730519386931635, 1393284965610134, 1597143735726030, 416032382447158, 1429665248828629 } + }, + { + { 360275475604565, 547835731063078, 215360904187529, 596646739879007, 332709650425085 }, + { 47602113726801, 1522314509708010, 437706261372925, 814035330438027, 335930650933545 }, + { 1291597595523886, 1058020588994081, 402837842324045, 1363323695882781, 2105763393033193 } + } +}, +{ /* 12/31 */ + { + { 109521982566564, 1715257748585139, 1112231216891516, 2046641005101484, 134249157157013 }, + { 2156991030936798, 2227544497153325, 1869050094431622, 754875860479115, 1754242344267058 }, + { 1846089562873800, 98894784984326, 1412430299204844, 171351226625762, 1100604760929008 } + }, + { + { 84172382130492, 499710970700046, 425749630620778, 1762872794206857, 612842602127960 }, + { 868309334532756, 1703010512741873, 1952690008738057, 4325269926064, 2071083554962116 }, + { 523094549451158, 401938899487815, 1407690589076010, 2022387426254453, 158660516411257 } + }, + { + { 612867287630009, 448212612103814, 571629077419196, 1466796750919376, 1728478129663858 }, + { 1723848973783452, 2208822520534681, 1718748322776940, 1974268454121942, 1194212502258141 }, + { 1254114807944608, 977770684047110, 2010756238954993, 1783628927194099, 1525962994408256 } + }, + { + { 232464058235826, 1948628555342434, 1835348780427694, 1031609499437291, 64472106918373 }, + { 767338676040683, 754089548318405, 1523192045639075, 435746025122062, 512692508440385 }, + { 1255955808701983, 1700487367990941, 1166401238800299, 1175121994891534, 1190934801395380 } + }, + { + { 349144008168292, 1337012557669162, 1475912332999108, 1321618454900458, 47611291904320 }, + { 877519947135419, 2172838026132651, 272304391224129, 1655143327559984, 886229406429814 }, + { 375806028254706, 214463229793940, 572906353144089, 572168269875638, 697556386112979 } + }, + { + { 1168827102357844, 823864273033637, 2071538752104697, 788062026895924, 599578340743362 }, + { 1948116082078088, 2054898304487796, 2204939184983900, 210526805152138, 786593586607626 }, + { 1915320147894736, 156481169009469, 655050471180417, 592917090415421, 2165897438660879 } + }, + { + { 1726336468579724, 1119932070398949, 1929199510967666, 33918788322959, 1836837863503150 }, + { 829996854845988, 217061778005138, 1686565909803640, 1346948817219846, 1723823550730181 }, + { 384301494966394, 687038900403062, 2211195391021739, 254684538421383, 1245698430589680 } + }, + { + { 1247567493562688, 1978182094455847, 183871474792955, 806570235643435, 288461518067916 }, + { 1449077384734201, 38285445457996, 2136537659177832, 2146493000841573, 725161151123125 }, + { 1201928866368855, 800415690605445, 1703146756828343, 997278587541744, 1858284414104014 } + } +}, +{ /* 13/31 */ + { + { 356468809648877, 782373916933152, 1718002439402870, 1392222252219254, 663171266061951 }, + { 759628738230460, 1012693474275852, 353780233086498, 246080061387552, 2030378857679162 }, + { 2040672435071076, 888593182036908, 1298443657189359, 1804780278521327, 354070726137060 } + }, + { + { 1894938527423184, 1463213041477277, 474410505497651, 247294963033299, 877975941029128 }, + { 207937160991127, 12966911039119, 820997788283092, 1010440472205286, 1701372890140810 }, + { 218882774543183, 533427444716285, 1233243976733245, 435054256891319, 1509568989549904 } + }, + { + { 1888838535711826, 1052177758340622, 1213553803324135, 169182009127332, 463374268115872 }, + { 299137589460312, 1594371588983567, 868058494039073, 257771590636681, 1805012993142921 }, + { 1806842755664364, 2098896946025095, 1356630998422878, 1458279806348064, 347755825962072 } + }, + { + { 1402334161391744, 1560083671046299, 1008585416617747, 1147797150908892, 1420416683642459 }, + { 665506704253369, 273770475169863, 799236974202630, 848328990077558, 1811448782807931 }, + { 1468412523962641, 771866649897997, 1931766110147832, 799561180078482, 524837559150077 } + }, + { + { 2223212657821850, 630416247363666, 2144451165500328, 816911130947791, 1024351058410032 }, + { 1266603897524861, 156378408858100, 1275649024228779, 447738405888420, 253186462063095 }, + { 2022215964509735, 136144366993649, 1800716593296582, 1193970603800203, 871675847064218 } + }, + { + { 1862751661970328, 851596246739884, 1519315554814041, 1542798466547449, 1417975335901520 }, + { 1228168094547481, 334133883362894, 587567568420081, 433612590281181, 603390400373205 }, + { 121893973206505, 1843345804916664, 1703118377384911, 497810164760654, 101150811654673 } + }, + { + { 458346255946468, 290909935619344, 1452768413850679, 550922875254215, 1537286854336538 }, + { 584322311184395, 380661238802118, 114839394528060, 655082270500073, 2111856026034852 }, + { 996965581008991, 2148998626477022, 1012273164934654, 1073876063914522, 1688031788934939 } + }, + { + { 923487018849600, 2085106799623355, 528082801620136, 1606206360876188, 735907091712524 }, + { 1697697887804317, 1335343703828273, 831288615207040, 949416685250051, 288760277392022 }, + { 1419122478109648, 1325574567803701, 602393874111094, 2107893372601700, 1314159682671307 } + } +}, +{ /* 14/31 */ + { + { 2201150872731804, 2180241023425241, 97663456423163, 1633405770247824, 848945042443986 }, + { 1173339555550611, 818605084277583, 47521504364289, 924108720564965, 735423405754506 }, + { 830104860549448, 1886653193241086, 1600929509383773, 1475051275443631, 286679780900937 } + }, + { + { 1577111294832995, 1030899169768747, 144900916293530, 1964672592979567, 568390100955250 }, + { 278388655910247, 487143369099838, 927762205508727, 181017540174210, 1616886700741287 }, + { 1191033906638969, 940823957346562, 1606870843663445, 861684761499847, 658674867251089 } + }, + { + { 1875032594195546, 1427106132796197, 724736390962158, 901860512044740, 635268497268760 }, + { 622869792298357, 1903919278950367, 1922588621661629, 1520574711600434, 1087100760174640 }, + { 25465949416618, 1693639527318811, 1526153382657203, 125943137857169, 145276964043999 } + }, + { + { 214739857969358, 920212862967915, 1939901550972269, 1211862791775221, 85097515720120 }, + { 2006245852772938, 734762734836159, 254642929763427, 1406213292755966, 239303749517686 }, + { 1619678837192149, 1919424032779215, 1357391272956794, 1525634040073113, 1310226789796241 } + }, + { + { 1040763709762123, 1704449869235352, 605263070456329, 1998838089036355, 1312142911487502 }, + { 1996723311435669, 1844342766567060, 985455700466044, 1165924681400960, 311508689870129 }, + { 43173156290518, 2202883069785309, 1137787467085917, 1733636061944606, 1394992037553852 } + }, + { + { 670078326344559, 555655025059356, 471959386282438, 2141455487356409, 849015953823125 }, + { 2197214573372804, 794254097241315, 1030190060513737, 267632515541902, 2040478049202624 }, + { 1812516004670529, 1609256702920783, 1706897079364493, 258549904773295, 996051247540686 } + }, + { + { 1540374301420584, 1764656898914615, 1810104162020396, 923808779163088, 664390074196579 }, + { 1323460699404750, 1262690757880991, 871777133477900, 1060078894988977, 1712236889662886 }, + { 1696163952057966, 1391710137550823, 608793846867416, 1034391509472039, 1780770894075012 } + }, + { + { 1367603834210841, 2131988646583224, 890353773628144, 1908908219165595, 270836895252891 }, + { 597536315471731, 40375058742586, 1942256403956049, 1185484645495932, 312666282024145 }, + { 1919411405316294, 1234508526402192, 1066863051997083, 1008444703737597, 1348810787701552 } + } +}, +{ /* 15/31 */ + { + { 2102881477513865, 1570274565945361, 1573617900503708, 18662635732583, 2232324307922098 }, + { 1853931367696942, 8107973870707, 350214504129299, 775206934582587, 1752317649166792 }, + { 1417148368003523, 721357181628282, 505725498207811, 373232277872983, 261634707184480 } + }, + { + { 2186733281493267, 2250694917008620, 1014829812957440, 479998161452389, 83566193876474 }, + { 1268116367301224, 560157088142809, 802626839600444, 2210189936605713, 1129993785579988 }, + { 615183387352312, 917611676109240, 878893615973325, 978940963313282, 938686890583575 } + }, + { + { 522024729211672, 1045059315315808, 1892245413707790, 1907891107684253, 2059998109500714 }, + { 1799679152208884, 912132775900387, 25967768040979, 432130448590461, 274568990261996 }, + { 98698809797682, 2144627600856209, 1907959298569602, 811491302610148, 1262481774981493 } + }, + { + { 1791451399743152, 1713538728337276, 118349997257490, 1882306388849954, 158235232210248 }, + { 1217809823321928, 2173947284933160, 1986927836272325, 1388114931125539, 12686131160169 }, + { 1650875518872272, 1136263858253897, 1732115601395988, 734312880662190, 1252904681142109 } + }, + { + { 372986456113865, 525430915458171, 2116279931702135, 501422713587815, 1907002872974925 }, + { 803147181835288, 868941437997146, 316299302989663, 943495589630550, 571224287904572 }, + { 227742695588364, 1776969298667369, 628602552821802, 457210915378118, 2041906378111140 } + }, + { + { 815000523470260, 913085688728307, 1052060118271173, 1345536665214223, 541623413135555 }, + { 1580216071604333, 1877997504342444, 857147161260913, 703522726778478, 2182763974211603 }, + { 1870080310923419, 71988220958492, 1783225432016732, 615915287105016, 1035570475990230 } + }, + { + { 730987750830150, 857613889540280, 1083813157271766, 1002817255970169, 1719228484436074 }, + { 377616581647602, 1581980403078513, 804044118130621, 2034382823044191, 643844048472185 }, + { 176957326463017, 1573744060478586, 528642225008045, 1816109618372371, 1515140189765006 } + }, + { + { 1888911448245718, 1387110895611080, 1924503794066429, 1731539523700949, 2230378382645454 }, + { 443392177002051, 233793396845137, 2199506622312416, 1011858706515937, 974676837063129 }, + { 1846351103143623, 1949984838808427, 671247021915253, 1946756846184401, 1929296930380217 } + } +}, +{ /* 16/31 */ + { + { 849646212452002, 1410198775302919, 73767886183695, 1641663456615812, 762256272452411 }, + { 692017667358279, 723305578826727, 1638042139863265, 748219305990306, 334589200523901 }, + { 22893968530686, 2235758574399251, 1661465835630252, 925707319443452, 1203475116966621 } + }, + { + { 801299035785166, 1733292596726131, 1664508947088596, 467749120991922, 1647498584535623 }, + { 903105258014366, 427141894933047, 561187017169777, 1884330244401954, 1914145708422219 }, + { 1344191060517578, 1960935031767890, 1518838929955259, 1781502350597190, 1564784025565682 } + }, + { + { 673723351748086, 1979969272514923, 1175287312495508, 1187589090978666, 1881897672213940 }, + { 1917185587363432, 1098342571752737, 5935801044414, 2000527662351839, 1538640296181569 }, + { 2495540013192, 678856913479236, 224998292422872, 219635787698590, 1972465269000940 } + }, + { + { 271413961212179, 1353052061471651, 344711291283483, 2014925838520662, 2006221033113941 }, + { 194583029968109, 514316781467765, 829677956235672, 1676415686873082, 810104584395840 }, + { 1980510813313589, 1948645276483975, 152063780665900, 129968026417582, 256984195613935 } + }, + { + { 1860190562533102, 1936576191345085, 461100292705964, 1811043097042830, 957486749306835 }, + { 796664815624365, 1543160838872951, 1500897791837765, 1667315977988401, 599303877030711 }, + { 1151480509533204, 2136010406720455, 738796060240027, 319298003765044, 1150614464349587 } + }, + { + { 1731069268103150, 735642447616087, 1364750481334268, 417232839982871, 927108269127661 }, + { 1017222050227968, 1987716148359, 2234319589635701, 621282683093392, 2132553131763026 }, + { 1567828528453324, 1017807205202360, 565295260895298, 829541698429100, 307243822276582 } + }, + { + { 249079270936248, 1501514259790706, 947909724204848, 944551802437487, 552658763982480 }, + { 2089966982947227, 1854140343916181, 2151980759220007, 2139781292261749, 158070445864917 }, + { 1338766321464554, 1906702607371284, 1519569445519894, 115384726262267, 1393058953390992 } + }, + { + { 1364621558265400, 1512388234908357, 1926731583198686, 2041482526432505, 920401122333774 }, + { 1884844597333588, 601480070269079, 620203503079537, 1079527400117915, 1202076693132015 }, + { 840922919763324, 727955812569642, 1303406629750194, 522898432152867, 294161410441865 } + } +}, +{ /* 17/31 */ + { + { 353760790835310, 1598361541848743, 1122905698202299, 1922533590158905, 419107700666580 }, + { 359856369838236, 180914355488683, 861726472646627, 218807937262986, 575626773232501 }, + { 755467689082474, 909202735047934, 730078068932500, 936309075711518, 2007798262842972 } + }, + { + { 1609384177904073, 362745185608627, 1335318541768201, 800965770436248, 547877979267412 }, + { 984339177776787, 815727786505884, 1645154585713747, 1659074964378553, 1686601651984156 }, + { 1697863093781930, 599794399429786, 1104556219769607, 830560774794755, 12812858601017 } + }, + { + { 1168737550514982, 897832437380552, 463140296333799, 302564600022547, 2008360505135501 }, + { 1856930662813910, 678090852002597, 1920179140755167, 1259527833759868, 55540971895511 }, + { 1158643631044921, 476554103621892, 178447851439725, 1305025542653569, 103433927680625 } + }, + { + { 2176793111709008, 1576725716350391, 2009350167273523, 2012390194631546, 2125297410909580 }, + { 825403285195098, 2144208587560784, 1925552004644643, 1915177840006985, 1015952128947864 }, + { 1807108316634472, 1534392066433717, 347342975407218, 1153820745616376, 7375003497471 } + }, + { + { 983061001799725, 431211889901241, 2201903782961093, 817393911064341, 2214616493042167 }, + { 228567918409756, 865093958780220, 358083886450556, 159617889659320, 1360637926292598 }, + { 234147501399755, 2229469128637390, 2175289352258889, 1397401514549353, 1885288963089922 } + }, + { + { 1111762412951562, 252849572507389, 1048714233823341, 146111095601446, 1237505378776770 }, + { 1113790697840279, 1051167139966244, 1045930658550944, 2011366241542643, 1686166824620755 }, + { 1054097349305049, 1872495070333352, 182121071220717, 1064378906787311, 100273572924182 } + }, + { + { 1306410853171605, 1627717417672447, 50983221088417, 1109249951172250, 870201789081392 }, + { 104233794644221, 1548919791188248, 2224541913267306, 2054909377116478, 1043803389015153 }, + { 216762189468802, 707284285441622, 190678557969733, 973969342604308, 1403009538434867 } + }, + { + { 1279024291038477, 344776835218310, 273722096017199, 1834200436811442, 634517197663804 }, + { 343805853118335, 1302216857414201, 566872543223541, 2051138939539004, 321428858384280 }, + { 470067171324852, 1618629234173951, 2000092177515639, 7307679772789, 1117521120249968 } + } +}, +{ /* 18/31 */ + { + { 278151578291475, 1810282338562947, 1771599529530998, 1383659409671631, 685373414471841 }, + { 577009397403102, 1791440261786291, 2177643735971638, 174546149911960, 1412505077782326 }, + { 893719721537457, 1201282458018197, 1522349501711173, 58011597740583, 1130406465887139 } + }, + { + { 412607348255453, 1280455764199780, 2233277987330768, 14180080401665, 331584698417165 }, + { 262483770854550, 990511055108216, 526885552771698, 571664396646158, 354086190278723 }, + { 1820352417585487, 24495617171480, 1547899057533253, 10041836186225, 480457105094042 } + }, + { + { 2023310314989233, 637905337525881, 2106474638900687, 557820711084072, 1687858215057826 }, + { 1144168702609745, 604444390410187, 1544541121756138, 1925315550126027, 626401428894002 }, + { 1922168257351784, 2018674099908659, 1776454117494445, 956539191509034, 36031129147635 } + }, + { + { 544644538748041, 1039872944430374, 876750409130610, 710657711326551, 1216952687484972 }, + { 58242421545916, 2035812695641843, 2118491866122923, 1191684463816273, 46921517454099 }, + { 272268252444639, 1374166457774292, 2230115177009552, 1053149803909880, 1354288411641016 } + }, + { + { 1857910905368338, 1754729879288912, 885945464109877, 1516096106802166, 1602902393369811 }, + { 1193437069800958, 901107149704790, 999672920611411, 477584824802207, 364239578697845 }, + { 886299989548838, 1538292895758047, 1590564179491896, 1944527126709657, 837344427345298 } + }, + { + { 754558365378305, 1712186480903618, 1703656826337531, 750310918489786, 518996040250900 }, + { 1309847803895382, 1462151862813074, 211370866671570, 1544595152703681, 1027691798954090 }, + { 803217563745370, 1884799722343599, 1357706345069218, 2244955901722095, 730869460037413 } + }, + { + { 689299471295966, 1831210565161071, 1375187341585438, 1106284977546171, 1893781834054269 }, + { 696351368613042, 1494385251239250, 738037133616932, 636385507851544, 927483222611406 }, + { 1949114198209333, 1104419699537997, 783495707664463, 1747473107602770, 2002634765788641 } + }, + { + { 1607325776830197, 530883941415333, 1451089452727895, 1581691157083423, 496100432831154 }, + { 1068900648804224, 2006891997072550, 1134049269345549, 1638760646180091, 2055396084625778 }, + { 2222475519314561, 1870703901472013, 1884051508440561, 1344072275216753, 1318025677799069 } + } +}, +{ /* 19/31 */ + { + { 155711679280656, 681100400509288, 389811735211209, 2135723811340709, 408733211204125 }, + { 7813206966729, 194444201427550, 2071405409526507, 1065605076176312, 1645486789731291 }, + { 16625790644959, 1647648827778410, 1579910185572704, 436452271048548, 121070048451050 } + }, + { + { 1037263028552531, 568385780377829, 297953104144430, 1558584511931211, 2238221839292471 }, + { 190565267697443, 672855706028058, 338796554369226, 337687268493904, 853246848691734 }, + { 1763863028400139, 766498079432444, 1321118624818005, 69494294452268, 858786744165651 } + }, + { + { 1292056768563024, 1456632109855638, 1100631247050184, 1386133165675321, 1232898350193752 }, + { 366253102478259, 525676242508811, 1449610995265438, 1183300845322183, 185960306491545 }, + { 28315355815982, 460422265558930, 1799675876678724, 1969256312504498, 1051823843138725 } + }, + { + { 156914999361983, 1606148405719949, 1665208410108430, 317643278692271, 1383783705665320 }, + { 54684536365732, 2210010038536222, 1194984798155308, 535239027773705, 1516355079301361 }, + { 1484387703771650, 198537510937949, 2186282186359116, 617687444857508, 647477376402122 } + }, + { + { 2147715541830533, 500032538445817, 646380016884826, 352227855331122, 1488268620408052 }, + { 159386186465542, 1877626593362941, 618737197060512, 1026674284330807, 1158121760792685 }, + { 1744544377739822, 1964054180355661, 1685781755873170, 2169740670377448, 1286112621104591 } + }, + { + { 81977249784993, 1667943117713086, 1668983819634866, 1605016835177615, 1353960708075544 }, + { 1602253788689063, 439542044889886, 2220348297664483, 657877410752869, 157451572512238 }, + { 1029287186166717, 65860128430192, 525298368814832, 1491902500801986, 1461064796385400 } + }, + { + { 408216988729246, 2121095722306989, 913562102267595, 1879708920318308, 241061448436731 }, + { 1185483484383269, 1356339572588553, 584932367316448, 102132779946470, 1792922621116791 }, + { 1966196870701923, 2230044620318636, 1425982460745905, 261167817826569, 46517743394330 } + }, + { + { 107077591595359, 884959942172345, 27306869797400, 2224911448949390, 964352058245223 }, + { 1730194207717538, 431790042319772, 1831515233279467, 1372080552768581, 1074513929381760 }, + { 1450880638731607, 1019861580989005, 1229729455116861, 1174945729836143, 826083146840706 } + } +}, +{ /* 20/31 */ + { + { 1899935429242705, 1602068751520477, 940583196550370, 82431069053859, 1540863155745696 }, + { 2136688454840028, 2099509000964294, 1690800495246475, 1217643678575476, 828720645084218 }, + { 765548025667841, 462473984016099, 998061409979798, 546353034089527, 2212508972466858 } + }, + { + { 46575283771160, 892570971573071, 1281983193144090, 1491520128287375, 75847005908304 }, + { 1801436127943107, 1734436817907890, 1268728090345068, 167003097070711, 2233597765834956 }, + { 1997562060465113, 1048700225534011, 7615603985628, 1855310849546841, 2242557647635213 } + }, + { + { 1161017320376250, 492624580169043, 2169815802355237, 976496781732542, 1770879511019629 }, + { 1357044908364776, 729130645262438, 1762469072918979, 1365633616878458, 181282906404941 }, + { 1080413443139865, 1155205815510486, 1848782073549786, 622566975152580, 124965574467971 } + }, + { + { 1184526762066993, 247622751762817, 692129017206356, 820018689412496, 2188697339828085 }, + { 2020536369003019, 202261491735136, 1053169669150884, 2056531979272544, 778165514694311 }, + { 237404399610207, 1308324858405118, 1229680749538400, 720131409105291, 1958958863624906 } + }, + { + { 515583508038846, 17656978857189, 1717918437373989, 1568052070792483, 46975803123923 }, + { 281527309158085, 36970532401524, 866906920877543, 2222282602952734, 1289598729589882 }, + { 1278207464902042, 494742455008756, 1262082121427081, 1577236621659884, 1888786707293291 } + }, + { + { 353042527954210, 1830056151907359, 1111731275799225, 174960955838824, 404312815582675 }, + { 2064251142068628, 1666421603389706, 1419271365315441, 468767774902855, 191535130366583 }, + { 1716987058588002, 1859366439773457, 1767194234188234, 64476199777924, 1117233614485261 } + }, + { + { 984292135520292, 135138246951259, 2220652137473167, 1722843421165029, 190482558012909 }, + { 298845952651262, 1166086588952562, 1179896526238434, 1347812759398693, 1412945390096208 }, + { 1143239552672925, 906436640714209, 2177000572812152, 2075299936108548, 325186347798433 } + }, + { + { 721024854374772, 684487861263316, 1373438744094159, 2193186935276995, 1387043709851261 }, + { 418098668140962, 715065997721283, 1471916138376055, 2168570337288357, 937812682637044 }, + { 1043584187226485, 2143395746619356, 2209558562919611, 482427979307092, 847556718384018 } + } +}, +{ /* 21/31 */ + { + { 1248731221520759, 1465200936117687, 540803492710140, 52978634680892, 261434490176109 }, + { 1057329623869501, 620334067429122, 461700859268034, 2012481616501857, 297268569108938 }, + { 1055352180870759, 1553151421852298, 1510903185371259, 1470458349428097, 1226259419062731 } + }, + { + { 1492988790301668, 790326625573331, 1190107028409745, 1389394752159193, 1620408196604194 }, + { 47000654413729, 1004754424173864, 1868044813557703, 173236934059409, 588771199737015 }, + { 30498470091663, 1082245510489825, 576771653181956, 806509986132686, 1317634017056939 } + }, + { + { 420308055751555, 1493354863316002, 165206721528088, 1884845694919786, 2065456951573059 }, + { 1115636332012334, 1854340990964155, 83792697369514, 1972177451994021, 457455116057587 }, + { 1698968457310898, 1435137169051090, 1083661677032510, 938363267483709, 340103887207182 } + }, + { + { 1995325341336574, 911500251774648, 164010755403692, 855378419194762, 1573601397528842 }, + { 241719380661528, 310028521317150, 1215881323380194, 1408214976493624, 2141142156467363 }, + { 1315157046163473, 727368447885818, 1363466668108618, 1668921439990361, 1398483384337907 } + }, + { + { 75029678299646, 1015388206460473, 1849729037055212, 1939814616452984, 444404230394954 }, + { 2053597130993710, 2024431685856332, 2233550957004860, 2012407275509545, 872546993104440 }, + { 1217269667678610, 599909351968693, 1390077048548598, 1471879360694802, 739586172317596 } + }, + { + { 1718318639380794, 1560510726633958, 904462881159922, 1418028351780052, 94404349451937 }, + { 2132502667405250, 214379346175414, 1502748313768060, 1960071701057800, 1353971822643138 }, + { 319394212043702, 2127459436033571, 717646691535162, 663366796076914, 318459064945314 } + }, + { + { 405989424923593, 1960452633787083, 667349034401665, 1492674260767112, 1451061489880787 }, + { 947085906234007, 323284730494107, 1485778563977200, 728576821512394, 901584347702286 }, + { 1575783124125742, 2126210792434375, 1569430791264065, 1402582372904727, 1891780248341114 } + }, + { + { 838432205560695, 1997703511451664, 1018791879907867, 1662001808174331, 78328132957753 }, + { 739152638255629, 2074935399403557, 505483666745895, 1611883356514088, 628654635394878 }, + { 1822054032121349, 643057948186973, 7306757352712, 577249257962099, 284735863382083 } + } +}, +{ /* 22/31 */ + { + { 1366558556363930, 1448606567552086, 1478881020944768, 165803179355898, 1115718458123498 }, + { 204146226972102, 1630511199034723, 2215235214174763, 174665910283542, 956127674017216 }, + { 1562934578796716, 1070893489712745, 11324610642270, 958989751581897, 2172552325473805 } + }, + { + { 1770564423056027, 735523631664565, 1326060113795289, 1509650369341127, 65892421582684 }, + { 623682558650637, 1337866509471512, 990313350206649, 1314236615762469, 1164772974270275 }, + { 223256821462517, 723690150104139, 1000261663630601, 933280913953265, 254872671543046 } + }, + { + { 1969087237026041, 624795725447124, 1335555107635969, 2069986355593023, 1712100149341902 }, + { 1236103475266979, 1837885883267218, 1026072585230455, 1025865513954973, 1801964901432134 }, + { 1115241013365517, 1712251818829143, 2148864332502771, 2096001471438138, 2235017246626125 } + }, + { + { 1299268198601632, 2047148477845621, 2165648650132450, 1612539282026145, 514197911628890 }, + { 118352772338543, 1067608711804704, 1434796676193498, 1683240170548391, 230866769907437 }, + { 1850689576796636, 1601590730430274, 1139674615958142, 1954384401440257, 76039205311 } + }, + { + { 1723387471374172, 997301467038410, 533927635123657, 20928644693965, 1756575222802513 }, + { 2146711623855116, 503278928021499, 625853062251406, 1109121378393107, 1033853809911861 }, + { 571005965509422, 2005213373292546, 1016697270349626, 56607856974274, 914438579435146 } + }, + { + { 1346698876211176, 2076651707527589, 1084761571110205, 265334478828406, 1068954492309671 }, + { 1769967932677654, 1695893319756416, 1151863389675920, 1781042784397689, 400287774418285 }, + { 1851867764003121, 403841933237558, 820549523771987, 761292590207581, 1743735048551143 } + }, + { + { 410915148140008, 2107072311871739, 1004367461876503, 99684895396761, 1180818713503224 }, + { 285945406881439, 648174397347453, 1098403762631981, 1366547441102991, 1505876883139217 }, + { 672095903120153, 1675918957959872, 636236529315028, 1569297300327696, 2164144194785875 } + }, + { + { 1902708175321798, 1035343530915438, 1178560808893263, 301095684058146, 1280977479761118 }, + { 1615357281742403, 404257611616381, 2160201349780978, 1160947379188955, 1578038619549541 }, + { 2013087639791217, 822734930507457, 1785668418619014, 1668650702946164, 389450875221715 } + } +}, +{ /* 23/31 */ + { + { 453918449698368, 106406819929001, 2072540975937135, 308588860670238, 1304394580755385 }, + { 1295082798350326, 2091844511495996, 1851348972587817, 3375039684596, 789440738712837 }, + { 2083069137186154, 848523102004566, 993982213589257, 1405313299916317, 1532824818698468 } + }, + { + { 1495961298852430, 1397203457344779, 1774950217066942, 139302743555696, 66603584342787 }, + { 1782411379088302, 1096724939964781, 27593390721418, 542241850291353, 1540337798439873 }, + { 693543956581437, 171507720360750, 1557908942697227, 1074697073443438, 1104093109037196 } + }, + { + { 345288228393419, 1099643569747172, 134881908403743, 1740551994106740, 248212179299770 }, + { 231429562203065, 1526290236421172, 2021375064026423, 1520954495658041, 806337791525116 }, + { 1079623667189886, 872403650198613, 766894200588288, 2163700860774109, 2023464507911816 } + }, + { + { 854645372543796, 1936406001954827, 151460662541253, 825325739271555, 1554306377287556 }, + { 1497138821904622, 1044820250515590, 1742593886423484, 1237204112746837, 849047450816987 }, + { 667962773375330, 1897271816877105, 1399712621683474, 1143302161683099, 2081798441209593 } + }, + { + { 127147851567005, 1936114012888110, 1704424366552046, 856674880716312, 716603621335359 }, + { 1072409664800960, 2146937497077528, 1508780108920651, 935767602384853, 1112800433544068 }, + { 333549023751292, 280219272863308, 2104176666454852, 1036466864875785, 536135186520207 } + }, + { + { 373666279883137, 146457241530109, 304116267127857, 416088749147715, 1258577131183391 }, + { 1186115062588401, 2251609796968486, 1098944457878953, 1153112761201374, 1791625503417267 }, + { 1870078460219737, 2129630962183380, 852283639691142, 292865602592851, 401904317342226 } + }, + { + { 1361070124828035, 815664541425524, 1026798897364671, 1951790935390647, 555874891834790 }, + { 1546301003424277, 459094500062839, 1097668518375311, 1780297770129643, 720763293687608 }, + { 1212405311403990, 1536693382542438, 61028431067459, 1863929423417129, 1223219538638038 } + }, + { + { 1294303766540260, 1183557465955093, 882271357233093, 63854569425375, 2213283684565087 }, + { 339050984211414, 601386726509773, 413735232134068, 966191255137228, 1839475899458159 }, + { 235605972169408, 2174055643032978, 1538335001838863, 1281866796917192, 1815940222628465 } + } +}, +{ /* 24/31 */ + { + { 1632352921721536, 1833328609514701, 2092779091951987, 1923956201873226, 2210068022482919 }, + { 35271216625062, 1712350667021807, 983664255668860, 98571260373038, 1232645608559836 }, + { 1998172393429622, 1798947921427073, 784387737563581, 1589352214827263, 1589861734168180 } + }, + { + { 1733739258725305, 31715717059538, 201969945218860, 992093044556990, 1194308773174556 }, + { 846415389605137, 746163495539180, 829658752826080, 592067705956946, 957242537821393 }, + { 1758148849754419, 619249044817679, 168089007997045, 1371497636330523, 1867101418880350 } + }, + { + { 326633984209635, 261759506071016, 1700682323676193, 1577907266349064, 1217647663383016 }, + { 1714182387328607, 1477856482074168, 574895689942184, 2159118410227270, 1555532449716575 }, + { 853828206885131, 998498946036955, 1835887550391235, 207627336608048, 258363815956050 } + }, + { + { 141141474651677, 1236728744905256, 643101419899887, 1646615130509173, 1208239602291765 }, + { 1501663228068911, 1354879465566912, 1444432675498247, 897812463852601, 855062598754348 }, + { 714380763546606, 1032824444965790, 1774073483745338, 1063840874947367, 1738680636537158 } + }, + { + { 1640635546696252, 633168953192112, 2212651044092396, 30590958583852, 368515260889378 }, + { 1171650314802029, 1567085444565577, 1453660792008405, 757914533009261, 1619511342778196 }, + { 420958967093237, 971103481109486, 2169549185607107, 1301191633558497, 1661514101014240 } + }, + { + { 907123651818302, 1332556122804146, 1824055253424487, 1367614217442959, 1982558335973172 }, + { 1121533090144639, 1021251337022187, 110469995947421, 1511059774758394, 2110035908131662 }, + { 303213233384524, 2061932261128138, 352862124777736, 40828818670255, 249879468482660 } + }, + { + { 856559257852200, 508517664949010, 1378193767894916, 1723459126947129, 1962275756614521 }, + { 1445691340537320, 40614383122127, 402104303144865, 485134269878232, 1659439323587426 }, + { 20057458979482, 1183363722525800, 2140003847237215, 2053873950687614, 2112017736174909 } + }, + { + { 2228654250927986, 1483591363415267, 1368661293910956, 1076511285177291, 526650682059608 }, + { 709481497028540, 531682216165724, 316963769431931, 1814315888453765, 258560242424104 }, + { 1053447823660455, 1955135194248683, 1010900954918985, 1182614026976701, 1240051576966610 } + } +}, +{ /* 25/31 */ + { + { 1957943897155497, 1788667368028035, 137692910029106, 1039519607062, 826404763313028 }, + { 1848942433095597, 1582009882530495, 1849292741020143, 1068498323302788, 2001402229799484 }, + { 1528282417624269, 2142492439828191, 2179662545816034, 362568973150328, 1591374675250271 } + }, + { + { 160026679434388, 232341189218716, 2149181472355545, 598041771119831, 183859001910173 }, + { 2013278155187349, 662660471354454, 793981225706267, 411706605985744, 804490933124791 }, + { 2051892037280204, 488391251096321, 2230187337030708, 930221970662692, 679002758255210 } + }, + { + { 1530723630438670, 875873929577927, 341560134269988, 449903119530753, 1055551308214179 }, + { 1461835919309432, 1955256480136428, 180866187813063, 1551979252664528, 557743861963950 }, + { 359179641731115, 1324915145732949, 902828372691474, 294254275669987, 1887036027752957 } + }, + { + { 2043271609454323, 2038225437857464, 1317528426475850, 1398989128982787, 2027639881006861 }, + { 2072902725256516, 312132452743412, 309930885642209, 996244312618453, 1590501300352303 }, + { 1397254305160710, 695734355138021, 2233992044438756, 1776180593969996, 1085588199351115 } + }, + { + { 440567051331029, 254894786356681, 493869224930222, 1556322069683366, 1567456540319218 }, + { 1950722461391320, 1907845598854797, 1822757481635527, 2121567704750244, 73811931471221 }, + { 387139307395758, 2058036430315676, 1220915649965325, 1794832055328951, 1230009312169328 } + }, + { + { 1765973779329517, 659344059446977, 19821901606666, 1301928341311214, 1116266004075885 }, + { 1127572801181483, 1224743760571696, 1276219889847274, 1529738721702581, 1589819666871853 }, + { 2181229378964934, 2190885205260020, 1511536077659137, 1246504208580490, 668883326494241 } + }, + { + { 437866655573314, 669026411194768, 81896997980338, 523874406393178, 245052060935236 }, + { 1975438052228868, 1071801519999806, 594652299224319, 1877697652668809, 1489635366987285 }, + { 958592545673770, 233048016518599, 851568750216589, 567703851596087, 1740300006094761 } + }, + { + { 2014540178270324, 192672779514432, 213877182641530, 2194819933853411, 1716422829364835 }, + { 1540769606609725, 2148289943846077, 1597804156127445, 1230603716683868, 815423458809453 }, + { 1738560251245018, 1779576754536888, 1783765347671392, 1880170990446751, 1088225159617541 } + } +}, +{ /* 26/31 */ + { + { 659303913929492, 1956447718227573, 1830568515922666, 841069049744408, 1669607124206368 }, + { 1143465490433355, 1532194726196059, 1093276745494697, 481041706116088, 2121405433561163 }, + { 1686424298744462, 1451806974487153, 266296068846582, 1834686947542675, 1720762336132256 } + }, + { + { 889217026388959, 1043290623284660, 856125087551909, 1669272323124636, 1603340330827879 }, + { 1206396181488998, 333158148435054, 1402633492821422, 1120091191722026, 1945474114550509 }, + { 766720088232571, 1512222781191002, 1189719893490790, 2091302129467914, 2141418006894941 } + }, + { + { 419663647306612, 1998875112167987, 1426599870253707, 1154928355379510, 486538532138187 }, + { 938160078005954, 1421776319053174, 1941643234741774, 180002183320818, 1414380336750546 }, + { 398001940109652, 1577721237663248, 1012748649830402, 1540516006905144, 1011684812884559 } + }, + { + { 1653276489969630, 6081825167624, 1921777941170836, 1604139841794531, 861211053640641 }, + { 996661541407379, 1455877387952927, 744312806857277, 139213896196746, 1000282908547789 }, + { 1450817495603008, 1476865707053229, 1030490562252053, 620966950353376, 1744760161539058 } + }, + { + { 559728410002599, 37056661641185, 2038622963352006, 1637244893271723, 1026565352238948 }, + { 962165956135846, 1116599660248791, 182090178006815, 1455605467021751, 196053588803284 }, + { 796863823080135, 1897365583584155, 420466939481601, 2165972651724672, 932177357788289 } + }, + { + { 877047233620632, 1375632631944375, 643773611882121, 660022738847877, 19353932331831 }, + { 2216943882299338, 394841323190322, 2222656898319671, 558186553950529, 1077236877025190 }, + { 801118384953213, 1914330175515892, 574541023311511, 1471123787903705, 1526158900256288 } + }, + { + { 949617889087234, 2207116611267331, 912920039141287, 501158539198789, 62362560771472 }, + { 1474518386765335, 1760793622169197, 1157399790472736, 1622864308058898, 165428294422792 }, + { 1961673048027128, 102619413083113, 1051982726768458, 1603657989805485, 1941613251499678 } + }, + { + { 1401939116319266, 335306339903072, 72046196085786, 862423201496006, 850518754531384 }, + { 1234706593321979, 1083343891215917, 898273974314935, 1640859118399498, 157578398571149 }, + { 1143483057726416, 1992614991758919, 674268662140796, 1773370048077526, 674318359920189 } + } +}, +{ /* 27/31 */ + { + { 1835401379538542, 173900035308392, 818247630716732, 1762100412152786, 1021506399448291 }, + { 1506632088156630, 2127481795522179, 513812919490255, 140643715928370, 442476620300318 }, + { 2056683376856736, 219094741662735, 2193541883188309, 1841182310235800, 556477468664293 } + }, + { + { 1315019427910827, 1049075855992603, 2066573052986543, 266904467185534, 2040482348591520 }, + { 94096246544434, 922482381166992, 24517828745563, 2139430508542503, 2097139044231004 }, + { 537697207950515, 1399352016347350, 1563663552106345, 2148749520888918, 549922092988516 } + }, + { + { 1747985413252434, 680511052635695, 1809559829982725, 594274250930054, 201673170745982 }, + { 323583936109569, 1973572998577657, 1192219029966558, 79354804385273, 1374043025560347 }, + { 213277331329947, 416202017849623, 1950535221091783, 1313441578103244, 2171386783823658 } + }, + { + { 189088804229831, 993969372859110, 895870121536987, 1547301535298256, 1477373024911350 }, + { 1620578418245010, 541035331188469, 2235785724453865, 2154865809088198, 1974627268751826 }, + { 1346805451740245, 1350981335690626, 942744349501813, 2155094562545502, 1012483751693409 } + }, + { + { 2107080134091762, 1132567062788208, 1824935377687210, 769194804343737, 1857941799971888 }, + { 1074666112436467, 249279386739593, 1174337926625354, 1559013532006480, 1472287775519121 }, + { 1872620123779532, 1892932666768992, 1921559078394978, 1270573311796160, 1438913646755037 } + }, + { + { 837390187648199, 1012253300223599, 989780015893987, 1351393287739814, 328627746545550 }, + { 1028328827183114, 1711043289969857, 1350832470374933, 1923164689604327, 1495656368846911 }, + { 1900828492104143, 430212361082163, 687437570852799, 832514536673512, 1685641495940794 } + }, + { + { 842632847936398, 605670026766216, 290836444839585, 163210774892356, 2213815011799645 }, + { 1176336383453996, 1725477294339771, 12700622672454, 678015708818208, 162724078519879 }, + { 1448049969043497, 1789411762943521, 385587766217753, 90201620913498, 832999441066823 } + }, + { + { 516086333293313, 2240508292484616, 1351669528166508, 1223255565316488, 750235824427138 }, + { 1263624896582495, 1102602401673328, 526302183714372, 2152015839128799, 1483839308490010 }, + { 442991718646863, 1599275157036458, 1925389027579192, 899514691371390, 350263251085160 } + } +}, +{ /* 28/31 */ + { + { 1689713572022143, 593854559254373, 978095044791970, 1985127338729499, 1676069120347625 }, + { 1557207018622683, 340631692799603, 1477725909476187, 614735951619419, 2033237123746766 }, + { 968764929340557, 1225534776710944, 662967304013036, 1155521416178595, 791142883466590 } + }, + { + { 1487081286167458, 993039441814934, 1792378982844640, 698652444999874, 2153908693179754 }, + { 1123181311102823, 685575944875442, 507605465509927, 1412590462117473, 568017325228626 }, + { 560258797465417, 2193971151466401, 1824086900849026, 579056363542056, 1690063960036441 } + }, + { + { 1918407319222416, 353767553059963, 1930426334528099, 1564816146005724, 1861342381708096 }, + { 2131325168777276, 1176636658428908, 1756922641512981, 1390243617176012, 1966325177038383 }, + { 2063958120364491, 2140267332393533, 699896251574968, 273268351312140, 375580724713232 } + }, + { + { 2024297515263178, 416959329722687, 1079014235017302, 171612225573183, 1031677520051053 }, + { 2033900009388450, 1744902869870788, 2190580087917640, 1949474984254121, 231049754293748 }, + { 343868674606581, 550155864008088, 1450580864229630, 481603765195050, 896972360018042 } + }, + { + { 2151139328380127, 314745882084928, 59756825775204, 1676664391494651, 2048348075599360 }, + { 1528930066340597, 1605003907059576, 1055061081337675, 1458319101947665, 1234195845213142 }, + { 830430507734812, 1780282976102377, 1425386760709037, 362399353095425, 2168861579799910 } + }, + { + { 1155762232730333, 980662895504006, 2053766700883521, 490966214077606, 510405877041357 }, + { 1683750316716132, 652278688286128, 1221798761193539, 1897360681476669, 319658166027343 }, + { 618808732869972, 72755186759744, 2060379135624181, 1730731526741822, 48862757828238 } + }, + { + { 1463171970593505, 1143040711767452, 614590986558883, 1409210575145591, 1882816996436803 }, + { 2230133264691131, 563950955091024, 2042915975426398, 827314356293472, 672028980152815 }, + { 264204366029760, 1654686424479449, 2185050199932931, 2207056159091748, 506015669043634 } + }, + { + { 1784446333136569, 1973746527984364, 334856327359575, 1156769775884610, 1023950124675478 }, + { 2065270940578383, 31477096270353, 306421879113491, 181958643936686, 1907105536686083 }, + { 1496516440779464, 1748485652986458, 872778352227340, 818358834654919, 97932669284220 } + } +}, +{ /* 29/31 */ + { + { 471636015770351, 672455402793577, 1804995246884103, 1842309243470804, 1501862504981682 }, + { 1013216974933691, 538921919682598, 1915776722521558, 1742822441583877, 1886550687916656 }, + { 2094270000643336, 303971879192276, 40801275554748, 649448917027930, 1818544418535447 } + }, + { + { 2241737709499165, 549397817447461, 838180519319392, 1725686958520781, 1705639080897747 }, + { 1216074541925116, 50120933933509, 1565829004133810, 721728156134580, 349206064666188 }, + { 948617110470858, 346222547451945, 1126511960599975, 1759386906004538, 493053284802266 } + }, + { + { 1454933046815146, 874696014266362, 1467170975468588, 1432316382418897, 2111710746366763 }, + { 2105387117364450, 1996463405126433, 1303008614294500, 851908115948209, 1353742049788635 }, + { 750300956351719, 1487736556065813, 15158817002104, 1511998221598392, 971739901354129 } + }, + { + { 1874648163531693, 2124487685930551, 1810030029384882, 918400043048335, 586348627300650 }, + { 1235084464747900, 1166111146432082, 1745394857881591, 1405516473883040, 4463504151617 }, + { 1663810156463827, 327797390285791, 1341846161759410, 1964121122800605, 1747470312055380 } + }, + { + { 660005247548233, 2071860029952887, 1358748199950107, 911703252219107, 1014379923023831 }, + { 2206641276178231, 1690587809721504, 1600173622825126, 2156096097634421, 1106822408548216 }, + { 1344788193552206, 1949552134239140, 1735915881729557, 675891104100469, 1834220014427292 } + }, + { + { 1920949492387964, 158885288387530, 70308263664033, 626038464897817, 1468081726101009 }, + { 622221042073383, 1210146474039168, 1742246422343683, 1403839361379025, 417189490895736 }, + { 22727256592983, 168471543384997, 1324340989803650, 1839310709638189, 504999476432775 } + }, + { + { 1313240518756327, 1721896294296942, 52263574587266, 2065069734239232, 804910473424630 }, + { 1337466662091884, 1287645354669772, 2018019646776184, 652181229374245, 898011753211715 }, + { 1969792547910734, 779969968247557, 2011350094423418, 1823964252907487, 1058949448296945 } + }, + { + { 207343737062002, 1118176942430253, 758894594548164, 806764629546266, 1157700123092949 }, + { 1273565321399022, 1638509681964574, 759235866488935, 666015124346707, 897983460943405 }, + { 1717263794012298, 1059601762860786, 1837819172257618, 1054130665797229, 680893204263559 } + } +}, +{ /* 30/31 */ + { + { 2237039662793603, 2249022333361206, 2058613546633703, 149454094845279, 2215176649164582 }, + { 79472182719605, 1851130257050174, 1825744808933107, 821667333481068, 781795293511946 }, + { 755822026485370, 152464789723500, 1178207602290608, 410307889503239, 156581253571278 } + }, + { + { 1418185496130297, 484520167728613, 1646737281442950, 1401487684670265, 1349185550126961 }, + { 1495380034400429, 325049476417173, 46346894893933, 1553408840354856, 828980101835683 }, + { 1280337889310282, 2070832742866672, 1640940617225222, 2098284908289951, 450929509534434 } + }, + { + { 407703353998781, 126572141483652, 286039827513621, 1999255076709338, 2030511179441770 }, + { 1254958221100483, 1153235960999843, 942907704968834, 637105404087392, 1149293270147267 }, + { 894249020470196, 400291701616810, 406878712230981, 1599128793487393, 1145868722604026 } + }, + { + { 1497955250203334, 110116344653260, 1128535642171976, 1900106496009660, 129792717460909 }, + { 452487513298665, 1352120549024569, 1173495883910956, 1999111705922009, 367328130454226 }, + { 1717539401269642, 1475188995688487, 891921989653942, 836824441505699, 1885988485608364 } + }, + { + { 1241784121422547, 187337051947583, 1118481812236193, 428747751936362, 30358898927325 }, + { 2022432361201842, 1088816090685051, 1977843398539868, 1854834215890724, 564238862029357 }, + { 938868489100585, 1100285072929025, 1017806255688848, 1957262154788833, 152787950560442 } + }, + { + { 867319417678923, 620471962942542, 226032203305716, 342001443957629, 1761675818237336 }, + { 1295072362439987, 931227904689414, 1355731432641687, 922235735834035, 892227229410209 }, + { 1680989767906154, 535362787031440, 2136691276706570, 1942228485381244, 1267350086882274 } + }, + { + { 366018233770527, 432660629755596, 126409707644535, 1973842949591662, 645627343442376 }, + { 535509430575217, 546885533737322, 1524675609547799, 2138095752851703, 1260738089896827 }, + { 1159906385590467, 2198530004321610, 714559485023225, 81880727882151, 1484020820037082 } + }, + { + { 1377485731340769, 2046328105512000, 1802058637158797, 62146136768173, 1356993908853901 }, + { 2013612215646735, 1830770575920375, 536135310219832, 609272325580394, 270684344495013 }, + { 1237542585982777, 2228682050256790, 1385281931622824, 593183794882890, 493654978552689 } + } +}, +{ /* 31/31 */ + { + { 47341488007760, 1891414891220257, 983894663308928, 176161768286818, 1126261115179708 }, + { 1694030170963455, 502038567066200, 1691160065225467, 949628319562187, 275110186693066 }, + { 1124515748676336, 1661673816593408, 1499640319059718, 1584929449166988, 558148594103306 } + }, + { + { 1784525599998356, 1619698033617383, 2097300287550715, 258265458103756, 1905684794832758 }, + { 1288941072872766, 931787902039402, 190731008859042, 2006859954667190, 1005931482221702 }, + { 1465551264822703, 152905080555927, 680334307368453, 173227184634745, 666407097159852 } + }, + { + { 2111017076203943, 1378760485794347, 1248583954016456, 1352289194864422, 1895180776543896 }, + { 171348223915638, 662766099800389, 462338943760497, 466917763340314, 656911292869115 }, + { 488623681976577, 866497561541722, 1708105560937768, 1673781214218839, 1506146329818807 } + }, + { + { 160425464456957, 950394373239689, 430497123340934, 711676555398832, 320964687779005 }, + { 988979367990485, 1359729327576302, 1301834257246029, 294141160829308, 29348272277475 }, + { 1434382743317910, 100082049942065, 221102347892623, 186982837860588, 1305765053501834 } + }, + { + { 2205916462268190, 499863829790820, 961960554686616, 158062762756985, 1841471168298305 }, + { 1191737341426592, 1847042034978363, 1382213545049056, 1039952395710448, 788812858896859 }, + { 1346965964571152, 1291881610839830, 2142916164336056, 786821641205979, 1571709146321039 } + }, + { + { 787164375951248, 202869205373189, 1356590421032140, 1431233331032510, 786341368775957 }, + { 492448143532951, 304105152670757, 1761767168301056, 233782684697790, 1981295323106089 }, + { 665807507761866, 1343384868355425, 895831046139653, 439338948736892, 1986828765695105 } + }, + { + { 756096210874553, 1721699973539149, 258765301727885, 1390588532210645, 1212530909934781 }, + { 852891097972275, 1816988871354562, 1543772755726524, 1174710635522444, 202129090724628 }, + { 1205281565824323, 22430498399418, 992947814485516, 1392458699738672, 688441466734558 } + }, + { + { 1050627428414972, 1955849529137135, 2171162376368357, 91745868298214, 447733118757826 }, + { 1287181461435438, 622722465530711, 880952150571872, 741035693459198, 311565274989772 }, + { 1003649078149734, 545233927396469, 1849786171789880, 1318943684880434, 280345687170552 } + } +} diff --git a/sodium/sodium_fe_51_base2.h b/sodium/sodium_fe_51_base2.h new file mode 100644 index 00000000..d0882416 --- /dev/null +++ b/sodium/sodium_fe_51_base2.h @@ -0,0 +1,40 @@ +{ + { 1288382639258501, 245678601348599, 269427782077623, 1462984067271730, 137412439391563 }, + { 62697248952638, 204681361388450, 631292143396476, 338455783676468, 1213667448819585 }, + { 301289933810280, 1259582250014073, 1422107436869536, 796239922652654, 1953934009299142 } +}, +{ + { 1601611775252272, 1720807796594148, 1132070835939856, 1260455018889551, 2147779492816911 }, + { 316559037616741, 2177824224946892, 1459442586438991, 1461528397712656, 751590696113597 }, + { 1850748884277385, 1200145853858453, 1068094770532492, 672251375690438, 1586055907191707 } +}, +{ + { 769950342298419, 132954430919746, 844085933195555, 974092374476333, 726076285546016 }, + { 425251763115706, 608463272472562, 442562545713235, 837766094556764, 374555092627893 }, + { 1086255230780037, 274979815921559, 1960002765731872, 929474102396301, 1190409889297339 } +}, +{ + { 665000864555967, 2065379846933859, 370231110385876, 350988370788628, 1233371373142985 }, + { 2019367628972465, 676711900706637, 110710997811333, 1108646842542025, 517791959672113 }, + { 965130719900578, 247011430587952, 526356006571389, 91986625355052, 2157223321444601 } +}, +{ + { 1802695059465007, 1664899123557221, 593559490740857, 2160434469266659, 927570450755031 }, + { 1725674970513508, 1933645953859181, 1542344539275782, 1767788773573747, 1297447965928905 }, + { 1381809363726107, 1430341051343062, 2061843536018959, 1551778050872521, 2036394857967624 } +}, +{ + { 1970894096313054, 528066325833207, 1619374932191227, 2207306624415883, 1169170329061080 }, + { 2070390218572616, 1458919061857835, 624171843017421, 1055332792707765, 433987520732508 }, + { 893653801273833, 1168026499324677, 1242553501121234, 1306366254304474, 1086752658510815 } +}, +{ + { 213454002618221, 939771523987438, 1159882208056014, 317388369627517, 621213314200687 }, + { 1971678598905747, 338026507889165, 762398079972271, 655096486107477, 42299032696322 }, + { 177130678690680, 1754759263300204, 1864311296286618, 1180675631479880, 1292726903152791 } +}, +{ + { 1913163449625248, 460779200291993, 2193883288642314, 1008900146920800, 1721983679009502 }, + { 1070401523076875, 1272492007800961, 1910153608563310, 2075579521696771, 1191169788841221 }, + { 692896803108118, 500174642072499, 2068223309439677, 1162190621851337, 1426986007309901 } +} diff --git a/sodium/sodium_fe_51_constants.h b/sodium/sodium_fe_51_constants.h new file mode 100644 index 00000000..53b75c6b --- /dev/null +++ b/sodium/sodium_fe_51_constants.h @@ -0,0 +1,21 @@ +/* 37095705934669439343138083508754565189542113879843219016388785533085940283555 */ +static const fe25519 d = { + 929955233495203, 466365720129213, 1662059464998953, 2033849074728123, 1442794654840575 +}; + +/* 2 * d = + * 16295367250680780974490674513165176452449235426866156013048779062215315747161 + */ +static const fe25519 d2 = { + 1859910466990425, 932731440258426, 1072319116312658, 1815898335770999, 633789495995903 +}; + +/* sqrt(-1) */ +static const fe25519 sqrtm1 = { + 1718705420411056, 234908883556509, 2233514472574048, 2117202627021982, 765476049583133 +}; + +/* A = 486662 */ +static const fe25519 curve25519_A = { + 486662, 0, 0, 0, 0 +}; diff --git a/sodium/sodium_fe_51_fe.h b/sodium/sodium_fe_51_fe.h new file mode 100644 index 00000000..de876264 --- /dev/null +++ b/sodium/sodium_fe_51_fe.h @@ -0,0 +1,116 @@ +/* + Ignores top bit of h. + */ + +void +fe25519_frombytes(fe25519 h, const unsigned char *s) +{ + const uint64_t mask = 0x7ffffffffffffULL; + uint64_t h0, h1, h2, h3, h4; + + h0 = (LOAD64_LE(s ) ) & mask; + h1 = (LOAD64_LE(s + 6) >> 3) & mask; + h2 = (LOAD64_LE(s + 12) >> 6) & mask; + h3 = (LOAD64_LE(s + 19) >> 1) & mask; + h4 = (LOAD64_LE(s + 24) >> 12) & mask; + + h[0] = h0; + h[1] = h1; + h[2] = h2; + h[3] = h3; + h[4] = h4; +} + +static void +fe25519_reduce(fe25519 h, const fe25519 f) +{ + const uint64_t mask = 0x7ffffffffffffULL; + uint128_t t[5]; + + t[0] = f[0]; + t[1] = f[1]; + t[2] = f[2]; + t[3] = f[3]; + t[4] = f[4]; + + t[1] += t[0] >> 51; + t[0] &= mask; + t[2] += t[1] >> 51; + t[1] &= mask; + t[3] += t[2] >> 51; + t[2] &= mask; + t[4] += t[3] >> 51; + t[3] &= mask; + t[0] += 19 * (t[4] >> 51); + t[4] &= mask; + + t[1] += t[0] >> 51; + t[0] &= mask; + t[2] += t[1] >> 51; + t[1] &= mask; + t[3] += t[2] >> 51; + t[2] &= mask; + t[4] += t[3] >> 51; + t[3] &= mask; + t[0] += 19 * (t[4] >> 51); + t[4] &= mask; + + /* now t is between 0 and 2^255-1, properly carried. */ + /* case 1: between 0 and 2^255-20. case 2: between 2^255-19 and 2^255-1. */ + + t[0] += 19ULL; + + t[1] += t[0] >> 51; + t[0] &= mask; + t[2] += t[1] >> 51; + t[1] &= mask; + t[3] += t[2] >> 51; + t[2] &= mask; + t[4] += t[3] >> 51; + t[3] &= mask; + t[0] += 19ULL * (t[4] >> 51); + t[4] &= mask; + + /* now between 19 and 2^255-1 in both cases, and offset by 19. */ + + t[0] += 0x8000000000000 - 19ULL; + t[1] += 0x8000000000000 - 1ULL; + t[2] += 0x8000000000000 - 1ULL; + t[3] += 0x8000000000000 - 1ULL; + t[4] += 0x8000000000000 - 1ULL; + + /* now between 2^255 and 2^256-20, and offset by 2^255. */ + + t[1] += t[0] >> 51; + t[0] &= mask; + t[2] += t[1] >> 51; + t[1] &= mask; + t[3] += t[2] >> 51; + t[2] &= mask; + t[4] += t[3] >> 51; + t[3] &= mask; + t[4] &= mask; + + h[0] = t[0]; + h[1] = t[1]; + h[2] = t[2]; + h[3] = t[3]; + h[4] = t[4]; +} + +void +fe25519_tobytes(unsigned char *s, const fe25519 h) +{ + fe25519 t; + uint64_t t0, t1, t2, t3; + + fe25519_reduce(t, h); + t0 = t[0] | (t[1] << 51); + t1 = (t[1] >> 13) | (t[2] << 38); + t2 = (t[2] >> 26) | (t[3] << 25); + t3 = (t[3] >> 39) | (t[4] << 12); + STORE64_LE(s + 0, t0); + STORE64_LE(s + 8, t1); + STORE64_LE(s + 16, t2); + STORE64_LE(s + 24, t3); +} diff --git a/sodium/sodium_generichash_blake2.c b/sodium/sodium_generichash_blake2.c new file mode 100644 index 00000000..a59592ec --- /dev/null +++ b/sodium/sodium_generichash_blake2.c @@ -0,0 +1,55 @@ +#include "sodium_crypto_generichash_blake2b.h" +#include "sodium_randombytes.h" + +size_t +crypto_generichash_blake2b_bytes_min(void) { + return crypto_generichash_blake2b_BYTES_MIN; +} + +size_t +crypto_generichash_blake2b_bytes_max(void) { + return crypto_generichash_blake2b_BYTES_MAX; +} + +size_t +crypto_generichash_blake2b_bytes(void) { + return crypto_generichash_blake2b_BYTES; +} + +size_t +crypto_generichash_blake2b_keybytes_min(void) { + return crypto_generichash_blake2b_KEYBYTES_MIN; +} + +size_t +crypto_generichash_blake2b_keybytes_max(void) { + return crypto_generichash_blake2b_KEYBYTES_MAX; +} + +size_t +crypto_generichash_blake2b_keybytes(void) { + return crypto_generichash_blake2b_KEYBYTES; +} + +size_t +crypto_generichash_blake2b_saltbytes(void) { + return crypto_generichash_blake2b_SALTBYTES; +} + +size_t +crypto_generichash_blake2b_personalbytes(void) { + return crypto_generichash_blake2b_PERSONALBYTES; +} + +size_t +crypto_generichash_blake2b_statebytes(void) +{ + return (sizeof(crypto_generichash_blake2b_state) + (size_t) 63U) + & ~(size_t) 63U; +} + +void +crypto_generichash_blake2b_keygen(unsigned char k[crypto_generichash_blake2b_KEYBYTES]) +{ + randombytes_buf(k, crypto_generichash_blake2b_KEYBYTES); +} diff --git a/sodium/sodium_generichash_blake2b.c b/sodium/sodium_generichash_blake2b.c new file mode 100644 index 00000000..a0949974 --- /dev/null +++ b/sodium/sodium_generichash_blake2b.c @@ -0,0 +1,117 @@ + +#include +#include +#include + +#include "sodium_blake2.h" +#include "sodium_crypto_generichash_blake2b.h" +#include "sodium_private_common.h" +#include "sodium_private_implementations.h" +#include + +int +crypto_generichash_blake2b(unsigned char *out, size_t outlen, + const unsigned char *in, unsigned long long inlen, + const unsigned char *key, size_t keylen) +{ + if (outlen <= 0U || outlen > BLAKE2B_OUTBYTES || + keylen > BLAKE2B_KEYBYTES || inlen > UINT64_MAX) { + return -1; + } + assert(outlen <= UINT8_MAX); + assert(keylen <= UINT8_MAX); + + return blake2b((uint8_t *) out, in, key, (uint8_t) outlen, (uint64_t) inlen, + (uint8_t) keylen); +} + +int +crypto_generichash_blake2b_salt_personal( + unsigned char *out, size_t outlen, const unsigned char *in, + unsigned long long inlen, const unsigned char *key, size_t keylen, + const unsigned char *salt, const unsigned char *personal) +{ + if (outlen <= 0U || outlen > BLAKE2B_OUTBYTES || + keylen > BLAKE2B_KEYBYTES || inlen > UINT64_MAX) { + return -1; + } + assert(outlen <= UINT8_MAX); + assert(keylen <= UINT8_MAX); + + return blake2b_salt_personal((uint8_t *) out, in, key, (uint8_t) outlen, + (uint64_t) inlen, (uint8_t) keylen, salt, + personal); +} + +int +crypto_generichash_blake2b_init(crypto_generichash_blake2b_state *state, + const unsigned char *key, const size_t keylen, + const size_t outlen) +{ + if (outlen <= 0U || outlen > BLAKE2B_OUTBYTES || + keylen > BLAKE2B_KEYBYTES) { + return -1; + } + assert(outlen <= UINT8_MAX); + assert(keylen <= UINT8_MAX); + COMPILER_ASSERT(sizeof(blake2b_state) <= sizeof *state); + if (key == NULL || keylen <= 0U) { + if (blake2b_init((blake2b_state *) (void *) state, (uint8_t) outlen) != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + } else if (blake2b_init_key((blake2b_state *) (void *) state, (uint8_t) outlen, key, + (uint8_t) keylen) != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + return 0; +} + +int +crypto_generichash_blake2b_init_salt_personal( + crypto_generichash_blake2b_state *state, const unsigned char *key, + const size_t keylen, const size_t outlen, const unsigned char *salt, + const unsigned char *personal) +{ + if (outlen <= 0U || outlen > BLAKE2B_OUTBYTES || + keylen > BLAKE2B_KEYBYTES) { + return -1; + } + assert(outlen <= UINT8_MAX); + assert(keylen <= UINT8_MAX); + if (key == NULL || keylen <= 0U) { + if (blake2b_init_salt_personal((blake2b_state *) (void *) state, + (uint8_t) outlen, salt, personal) != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + } else if (blake2b_init_key_salt_personal((blake2b_state *) (void *) state, + (uint8_t) outlen, key, + (uint8_t) keylen, salt, + personal) != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + return 0; +} + +int +crypto_generichash_blake2b_update(crypto_generichash_blake2b_state *state, + const unsigned char *in, + unsigned long long inlen) +{ + return blake2b_update((blake2b_state *) (void *) state, + (const uint8_t *) in, (uint64_t) inlen); +} + +int +crypto_generichash_blake2b_final(crypto_generichash_blake2b_state *state, + unsigned char *out, const size_t outlen) +{ + assert(outlen <= UINT8_MAX); + return blake2b_final((blake2b_state *) (void *) state, + (uint8_t *) out, (uint8_t) outlen); +} + +int +_crypto_generichash_blake2b_pick_best_implementation(void) +{ + return blake2b_pick_best_implementation(); +} diff --git a/sodium/sodium_hash_sha512.c b/sodium/sodium_hash_sha512.c new file mode 100644 index 00000000..7fb9a06c --- /dev/null +++ b/sodium/sodium_hash_sha512.c @@ -0,0 +1,13 @@ +#include "sodium_crypto_hash_sha512.h" + +size_t +crypto_hash_sha512_bytes(void) +{ + return crypto_hash_sha512_BYTES; +} + +size_t +crypto_hash_sha512_statebytes(void) +{ + return sizeof(crypto_hash_sha512_state); +} diff --git a/sodium/sodium_hash_sha512_cp.c b/sodium/sodium_hash_sha512_cp.c new file mode 100644 index 00000000..3ae32812 --- /dev/null +++ b/sodium/sodium_hash_sha512_cp.c @@ -0,0 +1,282 @@ + +/*- + * Copyright 2005,2007,2009 Colin Percival + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY + * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + * + */ + +#include +#include +#include +#include + +#include + +#include "sodium_crypto_hash_sha512.h" +#include "sodium_private_common.h" +#include "sodium_utils.h" + +static void +be64enc_vect(unsigned char *dst, const uint64_t *src, size_t len) +{ + size_t i; + + for (i = 0; i < len / 8; i++) { + STORE64_BE(dst + i * 8, src[i]); + } +} + +static void +be64dec_vect(uint64_t *dst, const unsigned char *src, size_t len) +{ + size_t i; + + for (i = 0; i < len / 8; i++) { + dst[i] = LOAD64_BE(src + i * 8); + } +} + +static const uint64_t Krnd[80] = { + 0x428a2f98d728ae22ULL, 0x7137449123ef65cdULL, 0xb5c0fbcfec4d3b2fULL, + 0xe9b5dba58189dbbcULL, 0x3956c25bf348b538ULL, 0x59f111f1b605d019ULL, + 0x923f82a4af194f9bULL, 0xab1c5ed5da6d8118ULL, 0xd807aa98a3030242ULL, + 0x12835b0145706fbeULL, 0x243185be4ee4b28cULL, 0x550c7dc3d5ffb4e2ULL, + 0x72be5d74f27b896fULL, 0x80deb1fe3b1696b1ULL, 0x9bdc06a725c71235ULL, + 0xc19bf174cf692694ULL, 0xe49b69c19ef14ad2ULL, 0xefbe4786384f25e3ULL, + 0x0fc19dc68b8cd5b5ULL, 0x240ca1cc77ac9c65ULL, 0x2de92c6f592b0275ULL, + 0x4a7484aa6ea6e483ULL, 0x5cb0a9dcbd41fbd4ULL, 0x76f988da831153b5ULL, + 0x983e5152ee66dfabULL, 0xa831c66d2db43210ULL, 0xb00327c898fb213fULL, + 0xbf597fc7beef0ee4ULL, 0xc6e00bf33da88fc2ULL, 0xd5a79147930aa725ULL, + 0x06ca6351e003826fULL, 0x142929670a0e6e70ULL, 0x27b70a8546d22ffcULL, + 0x2e1b21385c26c926ULL, 0x4d2c6dfc5ac42aedULL, 0x53380d139d95b3dfULL, + 0x650a73548baf63deULL, 0x766a0abb3c77b2a8ULL, 0x81c2c92e47edaee6ULL, + 0x92722c851482353bULL, 0xa2bfe8a14cf10364ULL, 0xa81a664bbc423001ULL, + 0xc24b8b70d0f89791ULL, 0xc76c51a30654be30ULL, 0xd192e819d6ef5218ULL, + 0xd69906245565a910ULL, 0xf40e35855771202aULL, 0x106aa07032bbd1b8ULL, + 0x19a4c116b8d2d0c8ULL, 0x1e376c085141ab53ULL, 0x2748774cdf8eeb99ULL, + 0x34b0bcb5e19b48a8ULL, 0x391c0cb3c5c95a63ULL, 0x4ed8aa4ae3418acbULL, + 0x5b9cca4f7763e373ULL, 0x682e6ff3d6b2b8a3ULL, 0x748f82ee5defb2fcULL, + 0x78a5636f43172f60ULL, 0x84c87814a1f0ab72ULL, 0x8cc702081a6439ecULL, + 0x90befffa23631e28ULL, 0xa4506cebde82bde9ULL, 0xbef9a3f7b2c67915ULL, + 0xc67178f2e372532bULL, 0xca273eceea26619cULL, 0xd186b8c721c0c207ULL, + 0xeada7dd6cde0eb1eULL, 0xf57d4f7fee6ed178ULL, 0x06f067aa72176fbaULL, + 0x0a637dc5a2c898a6ULL, 0x113f9804bef90daeULL, 0x1b710b35131c471bULL, + 0x28db77f523047d84ULL, 0x32caab7b40c72493ULL, 0x3c9ebe0a15c9bebcULL, + 0x431d67c49c100d4cULL, 0x4cc5d4becb3e42b6ULL, 0x597f299cfc657e2aULL, + 0x5fcb6fab3ad6faecULL, 0x6c44198c4a475817ULL +}; + +#define Ch(x, y, z) ((x & (y ^ z)) ^ z) +#define Maj(x, y, z) ((x & (y | z)) | (y & z)) +#define SHR(x, n) (x >> n) +#define ROTR(x, n) ROTR64(x, n) +#define S0(x) (ROTR(x, 28) ^ ROTR(x, 34) ^ ROTR(x, 39)) +#define S1(x) (ROTR(x, 14) ^ ROTR(x, 18) ^ ROTR(x, 41)) +#define s0(x) (ROTR(x, 1) ^ ROTR(x, 8) ^ SHR(x, 7)) +#define s1(x) (ROTR(x, 19) ^ ROTR(x, 61) ^ SHR(x, 6)) + +#define RND(a, b, c, d, e, f, g, h, k) \ + h += S1(e) + Ch(e, f, g) + k; \ + d += h; \ + h += S0(a) + Maj(a, b, c); + +#define RNDr(S, W, i, ii) \ + RND(S[(80 - i) % 8], S[(81 - i) % 8], S[(82 - i) % 8], S[(83 - i) % 8], \ + S[(84 - i) % 8], S[(85 - i) % 8], S[(86 - i) % 8], S[(87 - i) % 8], \ + W[i + ii] + Krnd[i + ii]) + +#define MSCH(W, ii, i) \ + W[i + ii + 16] = \ + s1(W[i + ii + 14]) + W[i + ii + 9] + s0(W[i + ii + 1]) + W[i + ii] + +static void +SHA512_Transform(uint64_t *state, const uint8_t block[128], uint64_t W[80], + uint64_t S[8]) +{ + int i; + + be64dec_vect(W, block, 128); + memcpy(S, state, 64); + for (i = 0; i < 80; i += 16) { + RNDr(S, W, 0, i); + RNDr(S, W, 1, i); + RNDr(S, W, 2, i); + RNDr(S, W, 3, i); + RNDr(S, W, 4, i); + RNDr(S, W, 5, i); + RNDr(S, W, 6, i); + RNDr(S, W, 7, i); + RNDr(S, W, 8, i); + RNDr(S, W, 9, i); + RNDr(S, W, 10, i); + RNDr(S, W, 11, i); + RNDr(S, W, 12, i); + RNDr(S, W, 13, i); + RNDr(S, W, 14, i); + RNDr(S, W, 15, i); + if (i == 64) { + break; + } + MSCH(W, 0, i); + MSCH(W, 1, i); + MSCH(W, 2, i); + MSCH(W, 3, i); + MSCH(W, 4, i); + MSCH(W, 5, i); + MSCH(W, 6, i); + MSCH(W, 7, i); + MSCH(W, 8, i); + MSCH(W, 9, i); + MSCH(W, 10, i); + MSCH(W, 11, i); + MSCH(W, 12, i); + MSCH(W, 13, i); + MSCH(W, 14, i); + MSCH(W, 15, i); + } + for (i = 0; i < 8; i++) { + state[i] += S[i]; + } +} + +static const uint8_t PAD[128] = { + 0x80, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 +}; + +static void +SHA512_Pad(crypto_hash_sha512_state *state, uint64_t tmp64[80 + 8]) +{ + unsigned int r; + unsigned int i; + + r = (unsigned int) ((state->count[1] >> 3) & 0x7f); + if (r < 112) { + for (i = 0; i < 112 - r; i++) { + state->buf[r + i] = PAD[i]; + } + } else { + for (i = 0; i < 128 - r; i++) { + state->buf[r + i] = PAD[i]; + } + SHA512_Transform(state->state, state->buf, &tmp64[0], &tmp64[80]); + memset(&state->buf[0], 0, 112); + } + be64enc_vect(&state->buf[112], state->count, 16); + SHA512_Transform(state->state, state->buf, &tmp64[0], &tmp64[80]); +} + +int +crypto_hash_sha512_init(crypto_hash_sha512_state *state) +{ + static const uint64_t sha512_initial_state[8] = { + 0x6a09e667f3bcc908ULL, 0xbb67ae8584caa73bULL, 0x3c6ef372fe94f82bULL, + 0xa54ff53a5f1d36f1ULL, 0x510e527fade682d1ULL, 0x9b05688c2b3e6c1fULL, + 0x1f83d9abfb41bd6bULL, 0x5be0cd19137e2179ULL + }; + + state->count[0] = state->count[1] = (uint64_t) 0U; + memcpy(state->state, sha512_initial_state, sizeof sha512_initial_state); + + return 0; +} + +int +crypto_hash_sha512_update(crypto_hash_sha512_state *state, + const unsigned char *in, unsigned long long inlen) +{ + uint64_t tmp64[80 + 8]; + uint64_t bitlen[2]; + unsigned long long i; + unsigned long long r; + + if (inlen <= 0U) { + return 0; + } + r = (unsigned long long) ((state->count[1] >> 3) & 0x7f); + + bitlen[1] = ((uint64_t) inlen) << 3; + bitlen[0] = ((uint64_t) inlen) >> 61; + /* LCOV_EXCL_START */ + if ((state->count[1] += bitlen[1]) < bitlen[1]) { + state->count[0]++; + } + /* LCOV_EXCL_STOP */ + state->count[0] += bitlen[0]; + if (inlen < 128 - r) { + for (i = 0; i < inlen; i++) { + state->buf[r + i] = in[i]; + } + return 0; + } + for (i = 0; i < 128 - r; i++) { + state->buf[r + i] = in[i]; + } + SHA512_Transform(state->state, state->buf, &tmp64[0], &tmp64[80]); + in += 128 - r; + inlen -= 128 - r; + + while (inlen >= 128) { + SHA512_Transform(state->state, in, &tmp64[0], &tmp64[80]); + in += 128; + inlen -= 128; + } + inlen &= 127; + for (i = 0; i < inlen; i++) { + state->buf[i] = in[i]; + } + sodium_memzero((void *) tmp64, sizeof tmp64); + + return 0; +} + +int +crypto_hash_sha512_final(crypto_hash_sha512_state *state, unsigned char *out) +{ + uint64_t tmp64[80 + 8]; + + SHA512_Pad(state, tmp64); + be64enc_vect(out, state->state, 64); + sodium_memzero((void *) tmp64, sizeof tmp64); + sodium_memzero((void *) state, sizeof *state); + + return 0; +} + +int +crypto_hash_sha512(unsigned char *out, const unsigned char *in, + unsigned long long inlen) +{ + crypto_hash_sha512_state state; + + crypto_hash_sha512_init(&state); + crypto_hash_sha512_update(&state, in, inlen); + crypto_hash_sha512_final(&state, out); + + return 0; +} diff --git a/sodium/sodium_onetimeauth_poly1305.c b/sodium/sodium_onetimeauth_poly1305.c new file mode 100644 index 00000000..18bfc0f6 --- /dev/null +++ b/sodium/sodium_onetimeauth_poly1305.c @@ -0,0 +1,97 @@ + +#include "sodium_onetimeauth_poly1305.h" +#include "sodium_crypto_onetimeauth_poly1305.h" +#include "sodium_private_common.h" +#include "sodium_private_implementations.h" +#include "sodium_randombytes.h" +#include "sodium_runtime.h" + +#include "sodium_poly1305_donna.h" +#if defined(HAVE_TI_MODE) && defined(HAVE_EMMINTRIN_H) +# include "sodium_poly1305-sse2.h" +#endif + +static const crypto_onetimeauth_poly1305_implementation *implementation = + &crypto_onetimeauth_poly1305_donna_implementation; + +int +crypto_onetimeauth_poly1305(unsigned char *out, const unsigned char *in, + unsigned long long inlen, const unsigned char *k) +{ + return implementation->onetimeauth(out, in, inlen, k); +} + +int +crypto_onetimeauth_poly1305_verify(const unsigned char *h, + const unsigned char *in, + unsigned long long inlen, + const unsigned char *k) +{ + return implementation->onetimeauth_verify(h, in, inlen, k); +} + +int +crypto_onetimeauth_poly1305_init(crypto_onetimeauth_poly1305_state *state, + const unsigned char *key) +{ + return implementation->onetimeauth_init(state, key); +} + +int +crypto_onetimeauth_poly1305_update(crypto_onetimeauth_poly1305_state *state, + const unsigned char *in, + unsigned long long inlen) +{ + return implementation->onetimeauth_update(state, in, inlen); +} + +int +crypto_onetimeauth_poly1305_final(crypto_onetimeauth_poly1305_state *state, + unsigned char *out) +{ + return implementation->onetimeauth_final(state, out); +} + +size_t +crypto_onetimeauth_poly1305_bytes(void) +{ + return crypto_onetimeauth_poly1305_BYTES; +} + +size_t +crypto_onetimeauth_poly1305_keybytes(void) +{ + return crypto_onetimeauth_poly1305_KEYBYTES; +} + +size_t +crypto_onetimeauth_poly1305_statebytes(void) +{ + return sizeof(crypto_onetimeauth_poly1305_state); +} + +void +crypto_onetimeauth_poly1305_keygen( + unsigned char k[crypto_onetimeauth_poly1305_KEYBYTES]) +{ + randombytes_buf(k, crypto_onetimeauth_poly1305_KEYBYTES); +} + +int +_crypto_onetimeauth_poly1305_pick_best_implementation(void) +{ + implementation = &crypto_onetimeauth_poly1305_donna_implementation; +#if defined(HAVE_TI_MODE) && defined(HAVE_EMMINTRIN_H) + if (sodium_runtime_has_sse2()) { + #if NETCODE_CRYPTO_LOGS + printf( "poly1305 -> sse3\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + implementation = &crypto_onetimeauth_poly1305_sse2_implementation; + return 0; + } +#endif + #if NETCODE_CRYPTO_LOGS + printf( "poly1305 -> ref\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + return 0; +} diff --git a/sodium/sodium_onetimeauth_poly1305.h b/sodium/sodium_onetimeauth_poly1305.h new file mode 100644 index 00000000..f914bad1 --- /dev/null +++ b/sodium/sodium_onetimeauth_poly1305.h @@ -0,0 +1,21 @@ + +#ifndef onetimeauth_poly1305_H +#define onetimeauth_poly1305_H + +#include "sodium_crypto_onetimeauth_poly1305.h" + +typedef struct crypto_onetimeauth_poly1305_implementation { + int (*onetimeauth)(unsigned char *out, const unsigned char *in, + unsigned long long inlen, const unsigned char *k); + int (*onetimeauth_verify)(const unsigned char *h, const unsigned char *in, + unsigned long long inlen, const unsigned char *k); + int (*onetimeauth_init)(crypto_onetimeauth_poly1305_state *state, + const unsigned char * key); + int (*onetimeauth_update)(crypto_onetimeauth_poly1305_state *state, + const unsigned char * in, + unsigned long long inlen); + int (*onetimeauth_final)(crypto_onetimeauth_poly1305_state *state, + unsigned char * out); +} crypto_onetimeauth_poly1305_implementation; + +#endif diff --git a/sodium/sodium_poly1305-sse2.c b/sodium/sodium_poly1305-sse2.c new file mode 100644 index 00000000..fe6efc05 --- /dev/null +++ b/sodium/sodium_poly1305-sse2.c @@ -0,0 +1,951 @@ + +#include +#include + +#include "sodium_onetimeauth_poly1305.h" +#include "sodium_crypto_verify_16.h" +#include "sodium_poly1305-sse2.h" +#include "sodium_private_common.h" +#include "sodium_private_sse2_64_32.h" +#include "sodium_utils.h" + +#if defined(HAVE_TI_MODE) && defined(HAVE_EMMINTRIN_H) + +# ifdef __GNUC__ +# pragma GCC target("sse2") +# endif + +# include + +typedef __m128i xmmi; + +# if defined(_MSC_VER) +# define POLY1305_NOINLINE __declspec(noinline) +# elif defined(__clang__) || defined(__GNUC__) +# define POLY1305_NOINLINE __attribute__((noinline)) +# else +# define POLY1305_NOINLINE +# endif + +# define poly1305_block_size 32 + +enum poly1305_state_flags_t { + poly1305_started = 1, + poly1305_final_shift8 = 4, + poly1305_final_shift16 = 8, + poly1305_final_r2_r = 16, /* use [r^2,r] for the final block */ + poly1305_final_r_1 = 32 /* use [r,1] for the final block */ +}; + +typedef struct poly1305_state_internal_t { + union { + uint64_t h[3]; + uint32_t hh[10]; + } H; /* 40 bytes */ + uint32_t R[5]; /* 20 bytes */ + uint32_t R2[5]; /* 20 bytes */ + uint32_t R4[5]; /* 20 bytes */ + uint64_t pad[2]; /* 16 bytes */ + uint64_t flags; /* 8 bytes */ + unsigned long long leftover; /* 8 bytes */ + unsigned char buffer[poly1305_block_size]; /* 32 bytes */ +} poly1305_state_internal_t; /* 164 bytes total */ + +/* + * _mm_loadl_epi64() is turned into a simple MOVQ. So, unaligned accesses are + * totally fine, even though this intrinsic requires a __m128i* input. + * This confuses dynamic analysis, so force alignment, only in debug mode. + */ +# ifdef DEBUG +static xmmi +_fakealign_mm_loadl_epi64(const void *m) +{ + xmmi tmp; + memcpy(&tmp, m, 8); + + return _mm_loadl_epi64(&tmp); +} +# define _mm_loadl_epi64(X) _fakealign_mm_loadl_epi64(X) +#endif + +/* copy 0-31 bytes */ +static inline void +poly1305_block_copy31(unsigned char *dst, const unsigned char *src, + unsigned long long bytes) +{ + if (bytes & 16) { + _mm_store_si128((xmmi *) (void *) dst, + _mm_loadu_si128((const xmmi *) (const void *) src)); + src += 16; + dst += 16; + } + if (bytes & 8) { + memcpy(dst, src, 8); + src += 8; + dst += 8; + } + if (bytes & 4) { + memcpy(dst, src, 4); + src += 4; + dst += 4; + } + if (bytes & 2) { + memcpy(dst, src, 2); + src += 2; + dst += 2; + } + if (bytes & 1) { + *dst = *src; + } +} + +static POLY1305_NOINLINE void +poly1305_init_ext(poly1305_state_internal_t *st, const unsigned char key[32], + unsigned long long bytes) +{ + uint32_t *R; + uint128_t d[3]; + uint64_t r0, r1, r2; + uint64_t rt0, rt1, rt2, st2, c; + uint64_t t0, t1; + unsigned long long i; + + if (!bytes) { + bytes = ~(unsigned long long) 0; + } + /* H = 0 */ + _mm_storeu_si128((xmmi *) (void *) &st->H.hh[0], _mm_setzero_si128()); + _mm_storeu_si128((xmmi *) (void *) &st->H.hh[4], _mm_setzero_si128()); + _mm_storeu_si128((xmmi *) (void *) &st->H.hh[8], _mm_setzero_si128()); + + /* clamp key */ + memcpy(&t0, key, 8); + memcpy(&t1, key + 8, 8); + r0 = t0 & 0xffc0fffffff; + t0 >>= 44; + t0 |= t1 << 20; + r1 = t0 & 0xfffffc0ffff; + t1 >>= 24; + r2 = t1 & 0x00ffffffc0f; + + /* r^1 */ + R = st->R; + R[0] = (uint32_t)(r0) &0x3ffffff; + R[1] = (uint32_t)((r0 >> 26) | (r1 << 18)) & 0x3ffffff; + R[2] = (uint32_t)((r1 >> 8)) & 0x3ffffff; + R[3] = (uint32_t)((r1 >> 34) | (r2 << 10)) & 0x3ffffff; + R[4] = (uint32_t)((r2 >> 16)); + + /* save pad */ + memcpy(&st->pad[0], key + 16, 8); + memcpy(&st->pad[1], key + 24, 8); + + rt0 = r0; + rt1 = r1; + rt2 = r2; + + /* r^2, r^4 */ + for (i = 0; i < 2; i++) { + if (i == 0) { + R = st->R2; + if (bytes <= 16) { + break; + } + } else if (i == 1) { + R = st->R4; + if (bytes < 96) { + break; + } + } + st2 = rt2 * (5 << 2); + + d[0] = ((uint128_t) rt0 * rt0) + ((uint128_t)(rt1 * 2) * st2); + d[1] = ((uint128_t) rt2 * st2) + ((uint128_t)(rt0 * 2) * rt1); + d[2] = ((uint128_t) rt1 * rt1) + ((uint128_t)(rt2 * 2) * rt0); + + rt0 = (uint64_t) d[0] & 0xfffffffffff; + c = (uint64_t)(d[0] >> 44); + d[1] += c; + + rt1 = (uint64_t) d[1] & 0xfffffffffff; + c = (uint64_t)(d[1] >> 44); + d[2] += c; + + rt2 = (uint64_t) d[2] & 0x3ffffffffff; + c = (uint64_t)(d[2] >> 42); + rt0 += c * 5; + c = (rt0 >> 44); + rt0 = rt0 & 0xfffffffffff; + rt1 += c; + c = (rt1 >> 44); + rt1 = rt1 & 0xfffffffffff; + rt2 += c; /* even if rt2 overflows, it will still fit in rp4 safely, and + is safe to multiply with */ + + R[0] = (uint32_t)(rt0) &0x3ffffff; + R[1] = (uint32_t)((rt0 >> 26) | (rt1 << 18)) & 0x3ffffff; + R[2] = (uint32_t)((rt1 >> 8)) & 0x3ffffff; + R[3] = (uint32_t)((rt1 >> 34) | (rt2 << 10)) & 0x3ffffff; + R[4] = (uint32_t)((rt2 >> 16)); + } + st->flags = 0; + st->leftover = 0U; +} + +static POLY1305_NOINLINE void +poly1305_blocks(poly1305_state_internal_t *st, const unsigned char *m, + unsigned long long bytes) +{ + CRYPTO_ALIGN(64) + xmmi HIBIT = + _mm_shuffle_epi32(_mm_cvtsi32_si128(1 << 24), _MM_SHUFFLE(1, 0, 1, 0)); + const xmmi MMASK = _mm_shuffle_epi32(_mm_cvtsi32_si128((1 << 26) - 1), + _MM_SHUFFLE(1, 0, 1, 0)); + const xmmi FIVE = + _mm_shuffle_epi32(_mm_cvtsi32_si128(5), _MM_SHUFFLE(1, 0, 1, 0)); + xmmi H0, H1, H2, H3, H4; + xmmi T0, T1, T2, T3, T4, T5, T6, T7, T8; + xmmi M0, M1, M2, M3, M4; + xmmi M5, M6, M7, M8; + xmmi C1, C2; + xmmi R20, R21, R22, R23, R24, S21, S22, S23, S24; + xmmi R40, R41, R42, R43, R44, S41, S42, S43, S44; + + if (st->flags & poly1305_final_shift8) { + HIBIT = _mm_srli_si128(HIBIT, 8); + } + if (st->flags & poly1305_final_shift16) { + HIBIT = _mm_setzero_si128(); + } + if (!(st->flags & poly1305_started)) { + /* H = [Mx,My] */ + T5 = _mm_unpacklo_epi64( + _mm_loadl_epi64((const xmmi *) (const void *) (m + 0)), + _mm_loadl_epi64((const xmmi *) (const void *) (m + 16))); + T6 = _mm_unpacklo_epi64( + _mm_loadl_epi64((const xmmi *) (const void *) (m + 8)), + _mm_loadl_epi64((const xmmi *) (const void *) (m + 24))); + H0 = _mm_and_si128(MMASK, T5); + H1 = _mm_and_si128(MMASK, _mm_srli_epi64(T5, 26)); + T5 = _mm_or_si128(_mm_srli_epi64(T5, 52), _mm_slli_epi64(T6, 12)); + H2 = _mm_and_si128(MMASK, T5); + H3 = _mm_and_si128(MMASK, _mm_srli_epi64(T5, 26)); + H4 = _mm_srli_epi64(T6, 40); + H4 = _mm_or_si128(H4, HIBIT); + m += 32; + bytes -= 32; + st->flags |= poly1305_started; + } else { + T0 = _mm_loadu_si128((const xmmi *) (const void *) &st->H.hh[0]); + T1 = _mm_loadu_si128((const xmmi *) (const void *) &st->H.hh[4]); + T2 = _mm_loadu_si128((const xmmi *) (const void *) &st->H.hh[8]); + H0 = _mm_shuffle_epi32(T0, _MM_SHUFFLE(1, 1, 0, 0)); + H1 = _mm_shuffle_epi32(T0, _MM_SHUFFLE(3, 3, 2, 2)); + H2 = _mm_shuffle_epi32(T1, _MM_SHUFFLE(1, 1, 0, 0)); + H3 = _mm_shuffle_epi32(T1, _MM_SHUFFLE(3, 3, 2, 2)); + H4 = _mm_shuffle_epi32(T2, _MM_SHUFFLE(1, 1, 0, 0)); + } + if (st->flags & (poly1305_final_r2_r | poly1305_final_r_1)) { + if (st->flags & poly1305_final_r2_r) { + /* use [r^2, r] */ + T2 = _mm_loadu_si128((const xmmi *) (const void *) &st->R[0]); + T3 = _mm_cvtsi32_si128(st->R[4]); + T0 = _mm_loadu_si128((const xmmi *) (const void *) &st->R2[0]); + T1 = _mm_cvtsi32_si128(st->R2[4]); + T4 = _mm_unpacklo_epi32(T0, T2); + T5 = _mm_unpackhi_epi32(T0, T2); + R24 = _mm_unpacklo_epi64(T1, T3); + } else { + /* use [r^1, 1] */ + T0 = _mm_loadu_si128((const xmmi *) (const void *) &st->R[0]); + T1 = _mm_cvtsi32_si128(st->R[4]); + T2 = _mm_cvtsi32_si128(1); + T4 = _mm_unpacklo_epi32(T0, T2); + T5 = _mm_unpackhi_epi32(T0, T2); + R24 = T1; + } + R20 = _mm_shuffle_epi32(T4, _MM_SHUFFLE(1, 1, 0, 0)); + R21 = _mm_shuffle_epi32(T4, _MM_SHUFFLE(3, 3, 2, 2)); + R22 = _mm_shuffle_epi32(T5, _MM_SHUFFLE(1, 1, 0, 0)); + R23 = _mm_shuffle_epi32(T5, _MM_SHUFFLE(3, 3, 2, 2)); + } else { + /* use [r^2, r^2] */ + T0 = _mm_loadu_si128((const xmmi *) (const void *) &st->R2[0]); + T1 = _mm_cvtsi32_si128(st->R2[4]); + R20 = _mm_shuffle_epi32(T0, _MM_SHUFFLE(0, 0, 0, 0)); + R21 = _mm_shuffle_epi32(T0, _MM_SHUFFLE(1, 1, 1, 1)); + R22 = _mm_shuffle_epi32(T0, _MM_SHUFFLE(2, 2, 2, 2)); + R23 = _mm_shuffle_epi32(T0, _MM_SHUFFLE(3, 3, 3, 3)); + R24 = _mm_shuffle_epi32(T1, _MM_SHUFFLE(0, 0, 0, 0)); + } + S21 = _mm_mul_epu32(R21, FIVE); + S22 = _mm_mul_epu32(R22, FIVE); + S23 = _mm_mul_epu32(R23, FIVE); + S24 = _mm_mul_epu32(R24, FIVE); + + if (bytes >= 64) { + T0 = _mm_loadu_si128((const xmmi *) (const void *) &st->R4[0]); + T1 = _mm_cvtsi32_si128(st->R4[4]); + R40 = _mm_shuffle_epi32(T0, _MM_SHUFFLE(0, 0, 0, 0)); + R41 = _mm_shuffle_epi32(T0, _MM_SHUFFLE(1, 1, 1, 1)); + R42 = _mm_shuffle_epi32(T0, _MM_SHUFFLE(2, 2, 2, 2)); + R43 = _mm_shuffle_epi32(T0, _MM_SHUFFLE(3, 3, 3, 3)); + R44 = _mm_shuffle_epi32(T1, _MM_SHUFFLE(0, 0, 0, 0)); + S41 = _mm_mul_epu32(R41, FIVE); + S42 = _mm_mul_epu32(R42, FIVE); + S43 = _mm_mul_epu32(R43, FIVE); + S44 = _mm_mul_epu32(R44, FIVE); + + while (bytes >= 64) { + xmmi v00, v01, v02, v03, v04; + xmmi v10, v11, v12, v13, v14; + xmmi v20, v21, v22, v23, v24; + xmmi v30, v31, v32, v33, v34; + xmmi v40, v41, v42, v43, v44; + xmmi T14, T15; + + /* H *= [r^4,r^4], preload [Mx,My] */ + T15 = S42; + T0 = H4; + T0 = _mm_mul_epu32(T0, S41); + v01 = H3; + v01 = _mm_mul_epu32(v01, T15); + T14 = S43; + T1 = H4; + T1 = _mm_mul_epu32(T1, T15); + v11 = H3; + v11 = _mm_mul_epu32(v11, T14); + T2 = H4; + T2 = _mm_mul_epu32(T2, T14); + T0 = _mm_add_epi64(T0, v01); + T15 = S44; + v02 = H2; + v02 = _mm_mul_epu32(v02, T14); + T3 = H4; + T3 = _mm_mul_epu32(T3, T15); + T1 = _mm_add_epi64(T1, v11); + v03 = H1; + v03 = _mm_mul_epu32(v03, T15); + v12 = H2; + v12 = _mm_mul_epu32(v12, T15); + T0 = _mm_add_epi64(T0, v02); + T14 = R40; + v21 = H3; + v21 = _mm_mul_epu32(v21, T15); + v31 = H3; + v31 = _mm_mul_epu32(v31, T14); + T0 = _mm_add_epi64(T0, v03); + T4 = H4; + T4 = _mm_mul_epu32(T4, T14); + T1 = _mm_add_epi64(T1, v12); + v04 = H0; + v04 = _mm_mul_epu32(v04, T14); + T2 = _mm_add_epi64(T2, v21); + v13 = H1; + v13 = _mm_mul_epu32(v13, T14); + T3 = _mm_add_epi64(T3, v31); + T15 = R41; + v22 = H2; + v22 = _mm_mul_epu32(v22, T14); + v32 = H2; + v32 = _mm_mul_epu32(v32, T15); + T0 = _mm_add_epi64(T0, v04); + v41 = H3; + v41 = _mm_mul_epu32(v41, T15); + T1 = _mm_add_epi64(T1, v13); + v14 = H0; + v14 = _mm_mul_epu32(v14, T15); + T2 = _mm_add_epi64(T2, v22); + T14 = R42; + T5 = _mm_unpacklo_epi64( + _mm_loadl_epi64((const xmmi *) (const void *) (m + 0)), + _mm_loadl_epi64((const xmmi *) (const void *) (m + 16))); + v23 = H1; + v23 = _mm_mul_epu32(v23, T15); + T3 = _mm_add_epi64(T3, v32); + v33 = H1; + v33 = _mm_mul_epu32(v33, T14); + T4 = _mm_add_epi64(T4, v41); + v42 = H2; + v42 = _mm_mul_epu32(v42, T14); + T1 = _mm_add_epi64(T1, v14); + T15 = R43; + T6 = _mm_unpacklo_epi64( + _mm_loadl_epi64((const xmmi *) (const void *) (m + 8)), + _mm_loadl_epi64((const xmmi *) (const void *) (m + 24))); + v24 = H0; + v24 = _mm_mul_epu32(v24, T14); + T2 = _mm_add_epi64(T2, v23); + v34 = H0; + v34 = _mm_mul_epu32(v34, T15); + T3 = _mm_add_epi64(T3, v33); + M0 = _mm_and_si128(MMASK, T5); + v43 = H1; + v43 = _mm_mul_epu32(v43, T15); + T4 = _mm_add_epi64(T4, v42); + M1 = _mm_and_si128(MMASK, _mm_srli_epi64(T5, 26)); + v44 = H0; + v44 = _mm_mul_epu32(v44, R44); + T2 = _mm_add_epi64(T2, v24); + T5 = _mm_or_si128(_mm_srli_epi64(T5, 52), _mm_slli_epi64(T6, 12)); + T3 = _mm_add_epi64(T3, v34); + M3 = _mm_and_si128(MMASK, _mm_srli_epi64(T6, 14)); + T4 = _mm_add_epi64(T4, v43); + M2 = _mm_and_si128(MMASK, T5); + T4 = _mm_add_epi64(T4, v44); + M4 = _mm_or_si128(_mm_srli_epi64(T6, 40), HIBIT); + + /* H += [Mx',My'] */ + T5 = _mm_loadu_si128((const xmmi *) (const void *) (m + 32)); + T6 = _mm_loadu_si128((const xmmi *) (const void *) (m + 48)); + T7 = _mm_unpacklo_epi32(T5, T6); + T8 = _mm_unpackhi_epi32(T5, T6); + M5 = _mm_unpacklo_epi32(T7, _mm_setzero_si128()); + M6 = _mm_unpackhi_epi32(T7, _mm_setzero_si128()); + M7 = _mm_unpacklo_epi32(T8, _mm_setzero_si128()); + M8 = _mm_unpackhi_epi32(T8, _mm_setzero_si128()); + M6 = _mm_slli_epi64(M6, 6); + M7 = _mm_slli_epi64(M7, 12); + M8 = _mm_slli_epi64(M8, 18); + T0 = _mm_add_epi64(T0, M5); + T1 = _mm_add_epi64(T1, M6); + T2 = _mm_add_epi64(T2, M7); + T3 = _mm_add_epi64(T3, M8); + T4 = _mm_add_epi64(T4, HIBIT); + + /* H += [Mx,My]*[r^2,r^2] */ + T15 = S22; + v00 = M4; + v00 = _mm_mul_epu32(v00, S21); + v01 = M3; + v01 = _mm_mul_epu32(v01, T15); + T14 = S23; + v10 = M4; + v10 = _mm_mul_epu32(v10, T15); + v11 = M3; + v11 = _mm_mul_epu32(v11, T14); + T0 = _mm_add_epi64(T0, v00); + v20 = M4; + v20 = _mm_mul_epu32(v20, T14); + T0 = _mm_add_epi64(T0, v01); + T15 = S24; + v02 = M2; + v02 = _mm_mul_epu32(v02, T14); + T1 = _mm_add_epi64(T1, v10); + v30 = M4; + v30 = _mm_mul_epu32(v30, T15); + T1 = _mm_add_epi64(T1, v11); + v03 = M1; + v03 = _mm_mul_epu32(v03, T15); + T2 = _mm_add_epi64(T2, v20); + v12 = M2; + v12 = _mm_mul_epu32(v12, T15); + T0 = _mm_add_epi64(T0, v02); + T14 = R20; + v21 = M3; + v21 = _mm_mul_epu32(v21, T15); + T3 = _mm_add_epi64(T3, v30); + v31 = M3; + v31 = _mm_mul_epu32(v31, T14); + T0 = _mm_add_epi64(T0, v03); + v40 = M4; + v40 = _mm_mul_epu32(v40, T14); + T1 = _mm_add_epi64(T1, v12); + v04 = M0; + v04 = _mm_mul_epu32(v04, T14); + T2 = _mm_add_epi64(T2, v21); + v13 = M1; + v13 = _mm_mul_epu32(v13, T14); + T3 = _mm_add_epi64(T3, v31); + T15 = R21; + v22 = M2; + v22 = _mm_mul_epu32(v22, T14); + T4 = _mm_add_epi64(T4, v40); + v32 = M2; + v32 = _mm_mul_epu32(v32, T15); + T0 = _mm_add_epi64(T0, v04); + v41 = M3; + v41 = _mm_mul_epu32(v41, T15); + T1 = _mm_add_epi64(T1, v13); + v14 = M0; + v14 = _mm_mul_epu32(v14, T15); + T2 = _mm_add_epi64(T2, v22); + T14 = R22; + v23 = M1; + v23 = _mm_mul_epu32(v23, T15); + T3 = _mm_add_epi64(T3, v32); + v33 = M1; + v33 = _mm_mul_epu32(v33, T14); + T4 = _mm_add_epi64(T4, v41); + v42 = M2; + v42 = _mm_mul_epu32(v42, T14); + T1 = _mm_add_epi64(T1, v14); + T15 = R23; + v24 = M0; + v24 = _mm_mul_epu32(v24, T14); + T2 = _mm_add_epi64(T2, v23); + v34 = M0; + v34 = _mm_mul_epu32(v34, T15); + T3 = _mm_add_epi64(T3, v33); + v43 = M1; + v43 = _mm_mul_epu32(v43, T15); + T4 = _mm_add_epi64(T4, v42); + v44 = M0; + v44 = _mm_mul_epu32(v44, R24); + T2 = _mm_add_epi64(T2, v24); + T3 = _mm_add_epi64(T3, v34); + T4 = _mm_add_epi64(T4, v43); + T4 = _mm_add_epi64(T4, v44); + + /* reduce */ + C1 = _mm_srli_epi64(T0, 26); + C2 = _mm_srli_epi64(T3, 26); + T0 = _mm_and_si128(T0, MMASK); + T3 = _mm_and_si128(T3, MMASK); + T1 = _mm_add_epi64(T1, C1); + T4 = _mm_add_epi64(T4, C2); + C1 = _mm_srli_epi64(T1, 26); + C2 = _mm_srli_epi64(T4, 26); + T1 = _mm_and_si128(T1, MMASK); + T4 = _mm_and_si128(T4, MMASK); + T2 = _mm_add_epi64(T2, C1); + T0 = _mm_add_epi64(T0, _mm_mul_epu32(C2, FIVE)); + C1 = _mm_srli_epi64(T2, 26); + C2 = _mm_srli_epi64(T0, 26); + T2 = _mm_and_si128(T2, MMASK); + T0 = _mm_and_si128(T0, MMASK); + T3 = _mm_add_epi64(T3, C1); + T1 = _mm_add_epi64(T1, C2); + C1 = _mm_srli_epi64(T3, 26); + T3 = _mm_and_si128(T3, MMASK); + T4 = _mm_add_epi64(T4, C1); + + /* Final: H = (H*[r^4,r^4] + [Mx,My]*[r^2,r^2] + [Mx',My']) */ + H0 = T0; + H1 = T1; + H2 = T2; + H3 = T3; + H4 = T4; + + m += 64; + bytes -= 64; + } + } + + if (bytes >= 32) { + xmmi v01, v02, v03, v04; + xmmi v11, v12, v13, v14; + xmmi v21, v22, v23, v24; + xmmi v31, v32, v33, v34; + xmmi v41, v42, v43, v44; + xmmi T14, T15; + + /* H *= [r^2,r^2] */ + T15 = S22; + T0 = H4; + T0 = _mm_mul_epu32(T0, S21); + v01 = H3; + v01 = _mm_mul_epu32(v01, T15); + T14 = S23; + T1 = H4; + T1 = _mm_mul_epu32(T1, T15); + v11 = H3; + v11 = _mm_mul_epu32(v11, T14); + T2 = H4; + T2 = _mm_mul_epu32(T2, T14); + T0 = _mm_add_epi64(T0, v01); + T15 = S24; + v02 = H2; + v02 = _mm_mul_epu32(v02, T14); + T3 = H4; + T3 = _mm_mul_epu32(T3, T15); + T1 = _mm_add_epi64(T1, v11); + v03 = H1; + v03 = _mm_mul_epu32(v03, T15); + v12 = H2; + v12 = _mm_mul_epu32(v12, T15); + T0 = _mm_add_epi64(T0, v02); + T14 = R20; + v21 = H3; + v21 = _mm_mul_epu32(v21, T15); + v31 = H3; + v31 = _mm_mul_epu32(v31, T14); + T0 = _mm_add_epi64(T0, v03); + T4 = H4; + T4 = _mm_mul_epu32(T4, T14); + T1 = _mm_add_epi64(T1, v12); + v04 = H0; + v04 = _mm_mul_epu32(v04, T14); + T2 = _mm_add_epi64(T2, v21); + v13 = H1; + v13 = _mm_mul_epu32(v13, T14); + T3 = _mm_add_epi64(T3, v31); + T15 = R21; + v22 = H2; + v22 = _mm_mul_epu32(v22, T14); + v32 = H2; + v32 = _mm_mul_epu32(v32, T15); + T0 = _mm_add_epi64(T0, v04); + v41 = H3; + v41 = _mm_mul_epu32(v41, T15); + T1 = _mm_add_epi64(T1, v13); + v14 = H0; + v14 = _mm_mul_epu32(v14, T15); + T2 = _mm_add_epi64(T2, v22); + T14 = R22; + v23 = H1; + v23 = _mm_mul_epu32(v23, T15); + T3 = _mm_add_epi64(T3, v32); + v33 = H1; + v33 = _mm_mul_epu32(v33, T14); + T4 = _mm_add_epi64(T4, v41); + v42 = H2; + v42 = _mm_mul_epu32(v42, T14); + T1 = _mm_add_epi64(T1, v14); + T15 = R23; + v24 = H0; + v24 = _mm_mul_epu32(v24, T14); + T2 = _mm_add_epi64(T2, v23); + v34 = H0; + v34 = _mm_mul_epu32(v34, T15); + T3 = _mm_add_epi64(T3, v33); + v43 = H1; + v43 = _mm_mul_epu32(v43, T15); + T4 = _mm_add_epi64(T4, v42); + v44 = H0; + v44 = _mm_mul_epu32(v44, R24); + T2 = _mm_add_epi64(T2, v24); + T3 = _mm_add_epi64(T3, v34); + T4 = _mm_add_epi64(T4, v43); + T4 = _mm_add_epi64(T4, v44); + + /* H += [Mx,My] */ + if (m) { + T5 = _mm_loadu_si128((const xmmi *) (const void *) (m + 0)); + T6 = _mm_loadu_si128((const xmmi *) (const void *) (m + 16)); + T7 = _mm_unpacklo_epi32(T5, T6); + T8 = _mm_unpackhi_epi32(T5, T6); + M0 = _mm_unpacklo_epi32(T7, _mm_setzero_si128()); + M1 = _mm_unpackhi_epi32(T7, _mm_setzero_si128()); + M2 = _mm_unpacklo_epi32(T8, _mm_setzero_si128()); + M3 = _mm_unpackhi_epi32(T8, _mm_setzero_si128()); + M1 = _mm_slli_epi64(M1, 6); + M2 = _mm_slli_epi64(M2, 12); + M3 = _mm_slli_epi64(M3, 18); + T0 = _mm_add_epi64(T0, M0); + T1 = _mm_add_epi64(T1, M1); + T2 = _mm_add_epi64(T2, M2); + T3 = _mm_add_epi64(T3, M3); + T4 = _mm_add_epi64(T4, HIBIT); + } + + /* reduce */ + C1 = _mm_srli_epi64(T0, 26); + C2 = _mm_srli_epi64(T3, 26); + T0 = _mm_and_si128(T0, MMASK); + T3 = _mm_and_si128(T3, MMASK); + T1 = _mm_add_epi64(T1, C1); + T4 = _mm_add_epi64(T4, C2); + C1 = _mm_srli_epi64(T1, 26); + C2 = _mm_srli_epi64(T4, 26); + T1 = _mm_and_si128(T1, MMASK); + T4 = _mm_and_si128(T4, MMASK); + T2 = _mm_add_epi64(T2, C1); + T0 = _mm_add_epi64(T0, _mm_mul_epu32(C2, FIVE)); + C1 = _mm_srli_epi64(T2, 26); + C2 = _mm_srli_epi64(T0, 26); + T2 = _mm_and_si128(T2, MMASK); + T0 = _mm_and_si128(T0, MMASK); + T3 = _mm_add_epi64(T3, C1); + T1 = _mm_add_epi64(T1, C2); + C1 = _mm_srli_epi64(T3, 26); + T3 = _mm_and_si128(T3, MMASK); + T4 = _mm_add_epi64(T4, C1); + + /* H = (H*[r^2,r^2] + [Mx,My]) */ + H0 = T0; + H1 = T1; + H2 = T2; + H3 = T3; + H4 = T4; + } + + if (m) { + T0 = _mm_shuffle_epi32(H0, _MM_SHUFFLE(0, 0, 2, 0)); + T1 = _mm_shuffle_epi32(H1, _MM_SHUFFLE(0, 0, 2, 0)); + T2 = _mm_shuffle_epi32(H2, _MM_SHUFFLE(0, 0, 2, 0)); + T3 = _mm_shuffle_epi32(H3, _MM_SHUFFLE(0, 0, 2, 0)); + T4 = _mm_shuffle_epi32(H4, _MM_SHUFFLE(0, 0, 2, 0)); + T0 = _mm_unpacklo_epi64(T0, T1); + T1 = _mm_unpacklo_epi64(T2, T3); + _mm_storeu_si128((xmmi *) (void *) &st->H.hh[0], T0); + _mm_storeu_si128((xmmi *) (void *) &st->H.hh[4], T1); + _mm_storel_epi64((xmmi *) (void *) &st->H.hh[8], T4); + } else { + uint32_t t0, t1, t2, t3, t4, b; + uint64_t h0, h1, h2, g0, g1, g2, c, nc; + + /* H = H[0]+H[1] */ + T0 = H0; + T1 = H1; + T2 = H2; + T3 = H3; + T4 = H4; + + T0 = _mm_add_epi64(T0, _mm_srli_si128(T0, 8)); + T1 = _mm_add_epi64(T1, _mm_srli_si128(T1, 8)); + T2 = _mm_add_epi64(T2, _mm_srli_si128(T2, 8)); + T3 = _mm_add_epi64(T3, _mm_srli_si128(T3, 8)); + T4 = _mm_add_epi64(T4, _mm_srli_si128(T4, 8)); + + t0 = _mm_cvtsi128_si32(T0); + b = (t0 >> 26); + t0 &= 0x3ffffff; + t1 = _mm_cvtsi128_si32(T1) + b; + b = (t1 >> 26); + t1 &= 0x3ffffff; + t2 = _mm_cvtsi128_si32(T2) + b; + b = (t2 >> 26); + t2 &= 0x3ffffff; + t3 = _mm_cvtsi128_si32(T3) + b; + b = (t3 >> 26); + t3 &= 0x3ffffff; + t4 = _mm_cvtsi128_si32(T4) + b; + + /* everything except t4 is in range, so this is all safe */ + h0 = (((uint64_t) t0) | ((uint64_t) t1 << 26)) & 0xfffffffffffull; + h1 = (((uint64_t) t1 >> 18) | ((uint64_t) t2 << 8) | + ((uint64_t) t3 << 34)) & + 0xfffffffffffull; + h2 = (((uint64_t) t3 >> 10) | ((uint64_t) t4 << 16)); + + c = (h2 >> 42); + h2 &= 0x3ffffffffff; + h0 += c * 5; + c = (h0 >> 44); + h0 &= 0xfffffffffff; + h1 += c; + c = (h1 >> 44); + h1 &= 0xfffffffffff; + h2 += c; + c = (h2 >> 42); + h2 &= 0x3ffffffffff; + h0 += c * 5; + c = (h0 >> 44); + h0 &= 0xfffffffffff; + h1 += c; + + g0 = h0 + 5; + c = (g0 >> 44); + g0 &= 0xfffffffffff; + g1 = h1 + c; + c = (g1 >> 44); + g1 &= 0xfffffffffff; + g2 = h2 + c - ((uint64_t) 1 << 42); + + c = (g2 >> 63) - 1; + nc = ~c; + h0 = (h0 & nc) | (g0 & c); + h1 = (h1 & nc) | (g1 & c); + h2 = (h2 & nc) | (g2 & c); + + st->H.h[0] = h0; + st->H.h[1] = h1; + st->H.h[2] = h2; + } +} + +static void +poly1305_update(poly1305_state_internal_t *st, const unsigned char *m, + unsigned long long bytes) +{ + unsigned long long i; + + /* handle leftover */ + if (st->leftover) { + unsigned long long want = (poly1305_block_size - st->leftover); + + if (want > bytes) { + want = bytes; + } + for (i = 0; i < want; i++) { + st->buffer[st->leftover + i] = m[i]; + } + bytes -= want; + m += want; + st->leftover += want; + if (st->leftover < poly1305_block_size) { + return; + } + poly1305_blocks(st, st->buffer, poly1305_block_size); + st->leftover = 0; + } + + /* process full blocks */ + if (bytes >= poly1305_block_size) { + unsigned long long want = (bytes & ~(poly1305_block_size - 1)); + + poly1305_blocks(st, m, want); + m += want; + bytes -= want; + } + + /* store leftover */ + if (bytes) { + for (i = 0; i < bytes; i++) { + st->buffer[st->leftover + i] = m[i]; + } + st->leftover += bytes; + } +} + +static POLY1305_NOINLINE void +poly1305_finish_ext(poly1305_state_internal_t *st, const unsigned char *m, + unsigned long long leftover, unsigned char mac[16]) +{ + uint64_t h0, h1, h2; + + if (leftover) { + CRYPTO_ALIGN(16) unsigned char final[32] = { 0 }; + + poly1305_block_copy31(final, m, leftover); + if (leftover != 16) { + final[leftover] = 1; + } + st->flags |= + (leftover >= 16) ? poly1305_final_shift8 : poly1305_final_shift16; + poly1305_blocks(st, final, 32); + } + + if (st->flags & poly1305_started) { + /* finalize, H *= [r^2,r], or H *= [r,1] */ + if (!leftover || (leftover > 16)) { + st->flags |= poly1305_final_r2_r; + } else { + st->flags |= poly1305_final_r_1; + } + poly1305_blocks(st, NULL, 32); + } + + h0 = st->H.h[0]; + h1 = st->H.h[1]; + h2 = st->H.h[2]; + + /* pad */ + h0 = ((h0) | (h1 << 44)); + h1 = ((h1 >> 20) | (h2 << 24)); +#ifdef HAVE_AMD64_ASM + __asm__ __volatile__( + "addq %2, %0 ;\n" + "adcq %3, %1 ;\n" + : "+r"(h0), "+r"(h1) + : "r"(st->pad[0]), "r"(st->pad[1]) + : "flags", "cc"); +#else + { + uint128_t h; + + memcpy(&h, &st->pad[0], 16); + h += ((uint128_t) h1 << 64) | h0; + h0 = (uint64_t) h; + h1 = (uint64_t)(h >> 64); + } +#endif + _mm_storeu_si128((xmmi *) (void *) st + 0, _mm_setzero_si128()); + _mm_storeu_si128((xmmi *) (void *) st + 1, _mm_setzero_si128()); + _mm_storeu_si128((xmmi *) (void *) st + 2, _mm_setzero_si128()); + _mm_storeu_si128((xmmi *) (void *) st + 3, _mm_setzero_si128()); + _mm_storeu_si128((xmmi *) (void *) st + 4, _mm_setzero_si128()); + _mm_storeu_si128((xmmi *) (void *) st + 5, _mm_setzero_si128()); + _mm_storeu_si128((xmmi *) (void *) st + 6, _mm_setzero_si128()); + _mm_storeu_si128((xmmi *) (void *) st + 7, _mm_setzero_si128()); + + memcpy(&mac[0], &h0, 8); + memcpy(&mac[8], &h1, 8); + + sodium_memzero((void *) st, sizeof *st); +} + +static void +poly1305_finish(poly1305_state_internal_t *st, unsigned char mac[16]) +{ + poly1305_finish_ext(st, st->buffer, st->leftover, mac); +} + +static int +crypto_onetimeauth_poly1305_sse2_init(crypto_onetimeauth_poly1305_state *state, + const unsigned char *key) +{ + COMPILER_ASSERT(sizeof(crypto_onetimeauth_poly1305_state) >= + sizeof(poly1305_state_internal_t)); + poly1305_init_ext((poly1305_state_internal_t *) (void *) state, key, 0U); + + return 0; +} + +static int +crypto_onetimeauth_poly1305_sse2_update( + crypto_onetimeauth_poly1305_state *state, const unsigned char *in, + unsigned long long inlen) +{ + poly1305_update((poly1305_state_internal_t *) (void *) state, in, inlen); + + return 0; +} + +static int +crypto_onetimeauth_poly1305_sse2_final(crypto_onetimeauth_poly1305_state *state, + unsigned char *out) +{ + poly1305_finish((poly1305_state_internal_t *) (void *) state, out); + + return 0; +} + +static int +crypto_onetimeauth_poly1305_sse2(unsigned char *out, const unsigned char *m, + unsigned long long inlen, + const unsigned char *key) +{ + CRYPTO_ALIGN(64) poly1305_state_internal_t st; + unsigned long long blocks; + + poly1305_init_ext(&st, key, inlen); + blocks = inlen & ~31; + if (blocks > 0) { + poly1305_blocks(&st, m, blocks); + m += blocks; + inlen -= blocks; + } + poly1305_finish_ext(&st, m, inlen, out); + + return 0; +} + +static int +crypto_onetimeauth_poly1305_sse2_verify(const unsigned char *h, + const unsigned char *in, + unsigned long long inlen, + const unsigned char *k) +{ + unsigned char correct[16]; + + crypto_onetimeauth_poly1305_sse2(correct, in, inlen, k); + + return crypto_verify_16(h, correct); +} + +struct crypto_onetimeauth_poly1305_implementation + crypto_onetimeauth_poly1305_sse2_implementation = { + SODIUM_C99(.onetimeauth =) crypto_onetimeauth_poly1305_sse2, + SODIUM_C99(.onetimeauth_verify =) + crypto_onetimeauth_poly1305_sse2_verify, + SODIUM_C99(.onetimeauth_init =) crypto_onetimeauth_poly1305_sse2_init, + SODIUM_C99(.onetimeauth_update =) + crypto_onetimeauth_poly1305_sse2_update, + SODIUM_C99(.onetimeauth_final =) crypto_onetimeauth_poly1305_sse2_final + }; + +#endif + +int poly1305_sse2_link_warning_dummy = 0; diff --git a/sodium/sodium_poly1305-sse2.h b/sodium/sodium_poly1305-sse2.h new file mode 100644 index 00000000..0bea92c6 --- /dev/null +++ b/sodium/sodium_poly1305-sse2.h @@ -0,0 +1,12 @@ +#ifndef poly1305_sse2_H +#define poly1305_sse2_H + +#include + +#include "sodium_onetimeauth_poly1305.h" +#include "sodium_crypto_onetimeauth_poly1305.h" + +extern struct crypto_onetimeauth_poly1305_implementation + crypto_onetimeauth_poly1305_sse2_implementation; + +#endif /* poly1305_sse2_H */ diff --git a/sodium/sodium_poly1305_donna.c b/sodium/sodium_poly1305_donna.c new file mode 100644 index 00000000..4891be43 --- /dev/null +++ b/sodium/sodium_poly1305_donna.c @@ -0,0 +1,124 @@ + +#include "sodium_poly1305_donna.h" +#include "sodium_crypto_verify_16.h" +#include "sodium_private_common.h" +#include "sodium_utils.h" + +#ifdef HAVE_TI_MODE +#include "sodium_poly1305_donna64.h" +#else +#include "sodium_poly1305_donna32.h" +#endif +#include "sodium_onetimeauth_poly1305.h" + +static void +poly1305_update(poly1305_state_internal_t *st, const unsigned char *m, + unsigned long long bytes) +{ + unsigned long long i; + + /* handle leftover */ + if (st->leftover) { + unsigned long long want = (poly1305_block_size - st->leftover); + + if (want > bytes) { + want = bytes; + } + for (i = 0; i < want; i++) { + st->buffer[st->leftover + i] = m[i]; + } + bytes -= want; + m += want; + st->leftover += want; + if (st->leftover < poly1305_block_size) { + return; + } + poly1305_blocks(st, st->buffer, poly1305_block_size); + st->leftover = 0; + } + + /* process full blocks */ + if (bytes >= poly1305_block_size) { + unsigned long long want = (bytes & ~(poly1305_block_size - 1)); + + poly1305_blocks(st, m, want); + m += want; + bytes -= want; + } + + /* store leftover */ + if (bytes) { + for (i = 0; i < bytes; i++) { + st->buffer[st->leftover + i] = m[i]; + } + st->leftover += bytes; + } +} + +static int +crypto_onetimeauth_poly1305_donna(unsigned char *out, const unsigned char *m, + unsigned long long inlen, + const unsigned char *key) +{ + CRYPTO_ALIGN(64) poly1305_state_internal_t state; + + poly1305_init(&state, key); + poly1305_update(&state, m, inlen); + poly1305_finish(&state, out); + + return 0; +} + +static int +crypto_onetimeauth_poly1305_donna_init(crypto_onetimeauth_poly1305_state *state, + const unsigned char *key) +{ + COMPILER_ASSERT(sizeof(crypto_onetimeauth_poly1305_state) >= + sizeof(poly1305_state_internal_t)); + poly1305_init((poly1305_state_internal_t *) (void *) state, key); + + return 0; +} + +static int +crypto_onetimeauth_poly1305_donna_update( + crypto_onetimeauth_poly1305_state *state, const unsigned char *in, + unsigned long long inlen) +{ + poly1305_update((poly1305_state_internal_t *) (void *) state, in, inlen); + + return 0; +} + +static int +crypto_onetimeauth_poly1305_donna_final( + crypto_onetimeauth_poly1305_state *state, unsigned char *out) +{ + poly1305_finish((poly1305_state_internal_t *) (void *) state, out); + + return 0; +} + +static int +crypto_onetimeauth_poly1305_donna_verify(const unsigned char *h, + const unsigned char *in, + unsigned long long inlen, + const unsigned char *k) +{ + unsigned char correct[16]; + + crypto_onetimeauth_poly1305_donna(correct, in, inlen, k); + + return crypto_verify_16(h, correct); +} + +struct crypto_onetimeauth_poly1305_implementation + crypto_onetimeauth_poly1305_donna_implementation = { + SODIUM_C99(.onetimeauth =) crypto_onetimeauth_poly1305_donna, + SODIUM_C99(.onetimeauth_verify =) + crypto_onetimeauth_poly1305_donna_verify, + SODIUM_C99(.onetimeauth_init =) crypto_onetimeauth_poly1305_donna_init, + SODIUM_C99(.onetimeauth_update =) + crypto_onetimeauth_poly1305_donna_update, + SODIUM_C99(.onetimeauth_final =) crypto_onetimeauth_poly1305_donna_final + }; diff --git a/sodium/sodium_poly1305_donna.h b/sodium/sodium_poly1305_donna.h new file mode 100644 index 00000000..9be44b89 --- /dev/null +++ b/sodium/sodium_poly1305_donna.h @@ -0,0 +1,12 @@ +#ifndef poly1305_donna_H +#define poly1305_donna_H + +#include + +#include "sodium_onetimeauth_poly1305.h" +#include "sodium_crypto_onetimeauth_poly1305.h" + +extern struct crypto_onetimeauth_poly1305_implementation + crypto_onetimeauth_poly1305_donna_implementation; + +#endif /* poly1305_donna_H */ diff --git a/sodium/sodium_poly1305_donna32.h b/sodium/sodium_poly1305_donna32.h new file mode 100644 index 00000000..fa723b73 --- /dev/null +++ b/sodium/sodium_poly1305_donna32.h @@ -0,0 +1,235 @@ +/* + poly1305 implementation using 32 bit * 32 bit = 64 bit multiplication + and 64 bit addition +*/ + +#if defined(_MSC_VER) +# define POLY1305_NOINLINE __declspec(noinline) +#elif defined(__clang__) || defined(__GNUC__) +# define POLY1305_NOINLINE __attribute__((noinline)) +#else +# define POLY1305_NOINLINE +#endif + +#include "sodium_private_common.h" + +#define poly1305_block_size 16 + +/* 17 + sizeof(unsigned long long) + 14*sizeof(unsigned long) */ +typedef struct poly1305_state_internal_t { + unsigned long r[5]; + unsigned long h[5]; + unsigned long pad[4]; + unsigned long long leftover; + unsigned char buffer[poly1305_block_size]; + unsigned char final; +} poly1305_state_internal_t; + +static void +poly1305_init(poly1305_state_internal_t *st, const unsigned char key[32]) +{ + /* r &= 0xffffffc0ffffffc0ffffffc0fffffff - wiped after finalization */ + st->r[0] = (LOAD32_LE(&key[0])) & 0x3ffffff; + st->r[1] = (LOAD32_LE(&key[3]) >> 2) & 0x3ffff03; + st->r[2] = (LOAD32_LE(&key[6]) >> 4) & 0x3ffc0ff; + st->r[3] = (LOAD32_LE(&key[9]) >> 6) & 0x3f03fff; + st->r[4] = (LOAD32_LE(&key[12]) >> 8) & 0x00fffff; + + /* h = 0 */ + st->h[0] = 0; + st->h[1] = 0; + st->h[2] = 0; + st->h[3] = 0; + st->h[4] = 0; + + /* save pad for later */ + st->pad[0] = LOAD32_LE(&key[16]); + st->pad[1] = LOAD32_LE(&key[20]); + st->pad[2] = LOAD32_LE(&key[24]); + st->pad[3] = LOAD32_LE(&key[28]); + + st->leftover = 0; + st->final = 0; +} + +static void +poly1305_blocks(poly1305_state_internal_t *st, const unsigned char *m, + unsigned long long bytes) +{ + const unsigned long hibit = (st->final) ? 0UL : (1UL << 24); /* 1 << 128 */ + unsigned long r0, r1, r2, r3, r4; + unsigned long s1, s2, s3, s4; + unsigned long h0, h1, h2, h3, h4; + unsigned long long d0, d1, d2, d3, d4; + unsigned long c; + + r0 = st->r[0]; + r1 = st->r[1]; + r2 = st->r[2]; + r3 = st->r[3]; + r4 = st->r[4]; + + s1 = r1 * 5; + s2 = r2 * 5; + s3 = r3 * 5; + s4 = r4 * 5; + + h0 = st->h[0]; + h1 = st->h[1]; + h2 = st->h[2]; + h3 = st->h[3]; + h4 = st->h[4]; + + while (bytes >= poly1305_block_size) { + /* h += m[i] */ + h0 += (LOAD32_LE(m + 0)) & 0x3ffffff; + h1 += (LOAD32_LE(m + 3) >> 2) & 0x3ffffff; + h2 += (LOAD32_LE(m + 6) >> 4) & 0x3ffffff; + h3 += (LOAD32_LE(m + 9) >> 6) & 0x3ffffff; + h4 += (LOAD32_LE(m + 12) >> 8) | hibit; + + /* h *= r */ + d0 = ((unsigned long long) h0 * r0) + ((unsigned long long) h1 * s4) + + ((unsigned long long) h2 * s3) + ((unsigned long long) h3 * s2) + + ((unsigned long long) h4 * s1); + d1 = ((unsigned long long) h0 * r1) + ((unsigned long long) h1 * r0) + + ((unsigned long long) h2 * s4) + ((unsigned long long) h3 * s3) + + ((unsigned long long) h4 * s2); + d2 = ((unsigned long long) h0 * r2) + ((unsigned long long) h1 * r1) + + ((unsigned long long) h2 * r0) + ((unsigned long long) h3 * s4) + + ((unsigned long long) h4 * s3); + d3 = ((unsigned long long) h0 * r3) + ((unsigned long long) h1 * r2) + + ((unsigned long long) h2 * r1) + ((unsigned long long) h3 * r0) + + ((unsigned long long) h4 * s4); + d4 = ((unsigned long long) h0 * r4) + ((unsigned long long) h1 * r3) + + ((unsigned long long) h2 * r2) + ((unsigned long long) h3 * r1) + + ((unsigned long long) h4 * r0); + + /* (partial) h %= p */ + c = (unsigned long) (d0 >> 26); + h0 = (unsigned long) d0 & 0x3ffffff; + d1 += c; + c = (unsigned long) (d1 >> 26); + h1 = (unsigned long) d1 & 0x3ffffff; + d2 += c; + c = (unsigned long) (d2 >> 26); + h2 = (unsigned long) d2 & 0x3ffffff; + d3 += c; + c = (unsigned long) (d3 >> 26); + h3 = (unsigned long) d3 & 0x3ffffff; + d4 += c; + c = (unsigned long) (d4 >> 26); + h4 = (unsigned long) d4 & 0x3ffffff; + h0 += c * 5; + c = (h0 >> 26); + h0 = h0 & 0x3ffffff; + h1 += c; + + m += poly1305_block_size; + bytes -= poly1305_block_size; + } + + st->h[0] = h0; + st->h[1] = h1; + st->h[2] = h2; + st->h[3] = h3; + st->h[4] = h4; +} + +static POLY1305_NOINLINE void +poly1305_finish(poly1305_state_internal_t *st, unsigned char mac[16]) +{ + unsigned long h0, h1, h2, h3, h4, c; + unsigned long g0, g1, g2, g3, g4; + unsigned long long f; + unsigned long mask; + + /* process the remaining block */ + if (st->leftover) { + unsigned long long i = st->leftover; + + st->buffer[i++] = 1; + for (; i < poly1305_block_size; i++) { + st->buffer[i] = 0; + } + st->final = 1; + poly1305_blocks(st, st->buffer, poly1305_block_size); + } + + /* fully carry h */ + h0 = st->h[0]; + h1 = st->h[1]; + h2 = st->h[2]; + h3 = st->h[3]; + h4 = st->h[4]; + + c = h1 >> 26; + h1 = h1 & 0x3ffffff; + h2 += c; + c = h2 >> 26; + h2 = h2 & 0x3ffffff; + h3 += c; + c = h3 >> 26; + h3 = h3 & 0x3ffffff; + h4 += c; + c = h4 >> 26; + h4 = h4 & 0x3ffffff; + h0 += c * 5; + c = h0 >> 26; + h0 = h0 & 0x3ffffff; + h1 += c; + + /* compute h + -p */ + g0 = h0 + 5; + c = g0 >> 26; + g0 &= 0x3ffffff; + g1 = h1 + c; + c = g1 >> 26; + g1 &= 0x3ffffff; + g2 = h2 + c; + c = g2 >> 26; + g2 &= 0x3ffffff; + g3 = h3 + c; + c = g3 >> 26; + g3 &= 0x3ffffff; + g4 = h4 + c - (1UL << 26); + + /* select h if h < p, or h + -p if h >= p */ + mask = (g4 >> ((sizeof(unsigned long) * 8) - 1)) - 1; + g0 &= mask; + g1 &= mask; + g2 &= mask; + g3 &= mask; + g4 &= mask; + mask = ~mask; + + h0 = (h0 & mask) | g0; + h1 = (h1 & mask) | g1; + h2 = (h2 & mask) | g2; + h3 = (h3 & mask) | g3; + h4 = (h4 & mask) | g4; + + /* h = h % (2^128) */ + h0 = ((h0) | (h1 << 26)) & 0xffffffff; + h1 = ((h1 >> 6) | (h2 << 20)) & 0xffffffff; + h2 = ((h2 >> 12) | (h3 << 14)) & 0xffffffff; + h3 = ((h3 >> 18) | (h4 << 8)) & 0xffffffff; + + /* mac = (h + pad) % (2^128) */ + f = (unsigned long long) h0 + st->pad[0]; + h0 = (unsigned long) f; + f = (unsigned long long) h1 + st->pad[1] + (f >> 32); + h1 = (unsigned long) f; + f = (unsigned long long) h2 + st->pad[2] + (f >> 32); + h2 = (unsigned long) f; + f = (unsigned long long) h3 + st->pad[3] + (f >> 32); + h3 = (unsigned long) f; + + STORE32_LE(mac + 0, (uint32_t) h0); + STORE32_LE(mac + 4, (uint32_t) h1); + STORE32_LE(mac + 8, (uint32_t) h2); + STORE32_LE(mac + 12, (uint32_t) h3); + + /* zero out the state */ + sodium_memzero((void *) st, sizeof *st); +} diff --git a/sodium/sodium_poly1305_donna64.h b/sodium/sodium_poly1305_donna64.h new file mode 100644 index 00000000..d2cdbf89 --- /dev/null +++ b/sodium/sodium_poly1305_donna64.h @@ -0,0 +1,220 @@ +/* + poly1305 implementation using 64 bit * 64 bit = 128 bit multiplication + and 128 bit addition +*/ + +#include "sodium_private_common.h" + +#define MUL(out, x, y) out = ((uint128_t) x * y) +#define ADD(out, in) out += in +#define ADDLO(out, in) out += in +#define SHR(in, shift) (unsigned long long) (in >> (shift)) +#define LO(in) (unsigned long long) (in) + +#if defined(_MSC_VER) +# define POLY1305_NOINLINE __declspec(noinline) +#elif defined(__clang__) || defined(__GNUC__) +# define POLY1305_NOINLINE __attribute__((noinline)) +#else +# define POLY1305_NOINLINE +#endif + +#define poly1305_block_size 16 + +/* 17 + sizeof(unsigned long long) + 8*sizeof(unsigned long long) */ +typedef struct poly1305_state_internal_t { + unsigned long long r[3]; + unsigned long long h[3]; + unsigned long long pad[2]; + unsigned long long leftover; + unsigned char buffer[poly1305_block_size]; + unsigned char final; +} poly1305_state_internal_t; + +static void +poly1305_init(poly1305_state_internal_t *st, const unsigned char key[32]) +{ + unsigned long long t0, t1; + + /* r &= 0xffffffc0ffffffc0ffffffc0fffffff */ + t0 = LOAD64_LE(&key[0]); + t1 = LOAD64_LE(&key[8]); + + /* wiped after finalization */ + st->r[0] = (t0) &0xffc0fffffff; + st->r[1] = ((t0 >> 44) | (t1 << 20)) & 0xfffffc0ffff; + st->r[2] = ((t1 >> 24)) & 0x00ffffffc0f; + + /* h = 0 */ + st->h[0] = 0; + st->h[1] = 0; + st->h[2] = 0; + + /* save pad for later */ + st->pad[0] = LOAD64_LE(&key[16]); + st->pad[1] = LOAD64_LE(&key[24]); + + st->leftover = 0; + st->final = 0; +} + +static void +poly1305_blocks(poly1305_state_internal_t *st, const unsigned char *m, + unsigned long long bytes) +{ + const unsigned long long hibit = + (st->final) ? 0ULL : (1ULL << 40); /* 1 << 128 */ + unsigned long long r0, r1, r2; + unsigned long long s1, s2; + unsigned long long h0, h1, h2; + unsigned long long c; + uint128_t d0, d1, d2, d; + + r0 = st->r[0]; + r1 = st->r[1]; + r2 = st->r[2]; + + h0 = st->h[0]; + h1 = st->h[1]; + h2 = st->h[2]; + + s1 = r1 * (5 << 2); + s2 = r2 * (5 << 2); + + while (bytes >= poly1305_block_size) { + unsigned long long t0, t1; + + /* h += m[i] */ + t0 = LOAD64_LE(&m[0]); + t1 = LOAD64_LE(&m[8]); + + h0 += ((t0) &0xfffffffffff); + h1 += (((t0 >> 44) | (t1 << 20)) & 0xfffffffffff); + h2 += (((t1 >> 24)) & 0x3ffffffffff) | hibit; + + /* h *= r */ + MUL(d0, h0, r0); + MUL(d, h1, s2); + ADD(d0, d); + MUL(d, h2, s1); + ADD(d0, d); + MUL(d1, h0, r1); + MUL(d, h1, r0); + ADD(d1, d); + MUL(d, h2, s2); + ADD(d1, d); + MUL(d2, h0, r2); + MUL(d, h1, r1); + ADD(d2, d); + MUL(d, h2, r0); + ADD(d2, d); + + /* (partial) h %= p */ + c = SHR(d0, 44); + h0 = LO(d0) & 0xfffffffffff; + ADDLO(d1, c); + c = SHR(d1, 44); + h1 = LO(d1) & 0xfffffffffff; + ADDLO(d2, c); + c = SHR(d2, 42); + h2 = LO(d2) & 0x3ffffffffff; + h0 += c * 5; + c = (h0 >> 44); + h0 = h0 & 0xfffffffffff; + h1 += c; + + m += poly1305_block_size; + bytes -= poly1305_block_size; + } + + st->h[0] = h0; + st->h[1] = h1; + st->h[2] = h2; +} + +static POLY1305_NOINLINE void +poly1305_finish(poly1305_state_internal_t *st, unsigned char mac[16]) +{ + unsigned long long h0, h1, h2, c; + unsigned long long g0, g1, g2; + unsigned long long t0, t1; + + /* process the remaining block */ + if (st->leftover) { + unsigned long long i = st->leftover; + + st->buffer[i] = 1; + + for (i = i + 1; i < poly1305_block_size; i++) { + st->buffer[i] = 0; + } + st->final = 1; + poly1305_blocks(st, st->buffer, poly1305_block_size); + } + + /* fully carry h */ + h0 = st->h[0]; + h1 = st->h[1]; + h2 = st->h[2]; + + c = (h1 >> 44); + h1 &= 0xfffffffffff; + h2 += c; + c = (h2 >> 42); + h2 &= 0x3ffffffffff; + h0 += c * 5; + c = (h0 >> 44); + h0 &= 0xfffffffffff; + h1 += c; + c = (h1 >> 44); + h1 &= 0xfffffffffff; + h2 += c; + c = (h2 >> 42); + h2 &= 0x3ffffffffff; + h0 += c * 5; + c = (h0 >> 44); + h0 &= 0xfffffffffff; + h1 += c; + + /* compute h + -p */ + g0 = h0 + 5; + c = (g0 >> 44); + g0 &= 0xfffffffffff; + g1 = h1 + c; + c = (g1 >> 44); + g1 &= 0xfffffffffff; + g2 = h2 + c - (1ULL << 42); + + /* select h if h < p, or h + -p if h >= p */ + c = (g2 >> ((sizeof(unsigned long long) * 8) - 1)) - 1; + g0 &= c; + g1 &= c; + g2 &= c; + c = ~c; + h0 = (h0 & c) | g0; + h1 = (h1 & c) | g1; + h2 = (h2 & c) | g2; + + /* h = (h + pad) */ + t0 = st->pad[0]; + t1 = st->pad[1]; + + h0 += ((t0) &0xfffffffffff); + c = (h0 >> 44); + h0 &= 0xfffffffffff; + h1 += (((t0 >> 44) | (t1 << 20)) & 0xfffffffffff) + c; + c = (h1 >> 44); + h1 &= 0xfffffffffff; + h2 += (((t1 >> 24)) & 0x3ffffffffff) + c; + h2 &= 0x3ffffffffff; + + /* mac = h % (2^128) */ + h0 = ((h0) | (h1 << 44)); + h1 = ((h1 >> 20) | (h2 << 24)); + + STORE64_LE(&mac[0], h0); + STORE64_LE(&mac[8], h1); + + /* zero out the state */ + sodium_memzero((void *) st, sizeof *st); +} diff --git a/sodium/sodium_private_chacha20_ietf_ext.h b/sodium/sodium_private_chacha20_ietf_ext.h new file mode 100644 index 00000000..2c80b96a --- /dev/null +++ b/sodium/sodium_private_chacha20_ietf_ext.h @@ -0,0 +1,16 @@ +#ifndef chacha20_ietf_ext_H +#define chacha20_ietf_ext_H + +#include + +/* The ietf_ext variant allows the internal counter to overflow into the IV */ + +int crypto_stream_chacha20_ietf_ext(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k); + +int crypto_stream_chacha20_ietf_ext_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, uint32_t ic, + const unsigned char *k); +#endif + diff --git a/sodium/sodium_private_common.h b/sodium/sodium_private_common.h new file mode 100644 index 00000000..6d569854 --- /dev/null +++ b/sodium/sodium_private_common.h @@ -0,0 +1,317 @@ +#ifndef common_H +#define common_H 1 + +#include +#include +#include + +#if defined( _MSC_VER ) +#pragma warning(disable:4127) +#pragma warning(disable:4244) +#pragma warning(disable:4668) +#pragma warning(disable:4456) +#endif + +#define COMPILER_ASSERT(X) (void) sizeof(char[(X) ? 1 : -1]) + +#ifdef HAVE_TI_MODE +# if defined(__SIZEOF_INT128__) +typedef unsigned __int128 uint128_t; +# else +typedef unsigned uint128_t __attribute__((mode(TI))); +# endif +#endif + +#define ROTL32(X, B) rotl32((X), (B)) +static inline uint32_t +rotl32(const uint32_t x, const int b) +{ + return (x << b) | (x >> (32 - b)); +} + +#define ROTL64(X, B) rotl64((X), (B)) +static inline uint64_t +rotl64(const uint64_t x, const int b) +{ + return (x << b) | (x >> (64 - b)); +} + +#define ROTR32(X, B) rotr32((X), (B)) +static inline uint32_t +rotr32(const uint32_t x, const int b) +{ + return (x >> b) | (x << (32 - b)); +} + +#define ROTR64(X, B) rotr64((X), (B)) +static inline uint64_t +rotr64(const uint64_t x, const int b) +{ + return (x >> b) | (x << (64 - b)); +} + +#define LOAD64_LE(SRC) load64_le(SRC) +static inline uint64_t +load64_le(const uint8_t src[8]) +{ +#ifdef NATIVE_LITTLE_ENDIAN + uint64_t w; + memcpy(&w, src, sizeof w); + return w; +#else + uint64_t w = (uint64_t) src[0]; + w |= (uint64_t) src[1] << 8; + w |= (uint64_t) src[2] << 16; + w |= (uint64_t) src[3] << 24; + w |= (uint64_t) src[4] << 32; + w |= (uint64_t) src[5] << 40; + w |= (uint64_t) src[6] << 48; + w |= (uint64_t) src[7] << 56; + return w; +#endif +} + +#define STORE64_LE(DST, W) store64_le((DST), (W)) +static inline void +store64_le(uint8_t dst[8], uint64_t w) +{ +#ifdef NATIVE_LITTLE_ENDIAN + memcpy(dst, &w, sizeof w); +#else + dst[0] = (uint8_t) w; w >>= 8; + dst[1] = (uint8_t) w; w >>= 8; + dst[2] = (uint8_t) w; w >>= 8; + dst[3] = (uint8_t) w; w >>= 8; + dst[4] = (uint8_t) w; w >>= 8; + dst[5] = (uint8_t) w; w >>= 8; + dst[6] = (uint8_t) w; w >>= 8; + dst[7] = (uint8_t) w; +#endif +} + +#define LOAD32_LE(SRC) load32_le(SRC) +static inline uint32_t +load32_le(const uint8_t src[4]) +{ +#ifdef NATIVE_LITTLE_ENDIAN + uint32_t w; + memcpy(&w, src, sizeof w); + return w; +#else + uint32_t w = (uint32_t) src[0]; + w |= (uint32_t) src[1] << 8; + w |= (uint32_t) src[2] << 16; + w |= (uint32_t) src[3] << 24; + return w; +#endif +} + +#define STORE32_LE(DST, W) store32_le((DST), (W)) +static inline void +store32_le(uint8_t dst[4], uint32_t w) +{ +#ifdef NATIVE_LITTLE_ENDIAN + memcpy(dst, &w, sizeof w); +#else + dst[0] = (uint8_t) w; w >>= 8; + dst[1] = (uint8_t) w; w >>= 8; + dst[2] = (uint8_t) w; w >>= 8; + dst[3] = (uint8_t) w; +#endif +} + +/* ----- */ + +#define LOAD64_BE(SRC) load64_be(SRC) +static inline uint64_t +load64_be(const uint8_t src[8]) +{ +#ifdef NATIVE_BIG_ENDIAN + uint64_t w; + memcpy(&w, src, sizeof w); + return w; +#else + uint64_t w = (uint64_t) src[7]; + w |= (uint64_t) src[6] << 8; + w |= (uint64_t) src[5] << 16; + w |= (uint64_t) src[4] << 24; + w |= (uint64_t) src[3] << 32; + w |= (uint64_t) src[2] << 40; + w |= (uint64_t) src[1] << 48; + w |= (uint64_t) src[0] << 56; + return w; +#endif +} + +#define STORE64_BE(DST, W) store64_be((DST), (W)) +static inline void +store64_be(uint8_t dst[8], uint64_t w) +{ +#ifdef NATIVE_BIG_ENDIAN + memcpy(dst, &w, sizeof w); +#else + dst[7] = (uint8_t) w; w >>= 8; + dst[6] = (uint8_t) w; w >>= 8; + dst[5] = (uint8_t) w; w >>= 8; + dst[4] = (uint8_t) w; w >>= 8; + dst[3] = (uint8_t) w; w >>= 8; + dst[2] = (uint8_t) w; w >>= 8; + dst[1] = (uint8_t) w; w >>= 8; + dst[0] = (uint8_t) w; +#endif +} + +#define LOAD32_BE(SRC) load32_be(SRC) +static inline uint32_t +load32_be(const uint8_t src[4]) +{ +#ifdef NATIVE_BIG_ENDIAN + uint32_t w; + memcpy(&w, src, sizeof w); + return w; +#else + uint32_t w = (uint32_t) src[3]; + w |= (uint32_t) src[2] << 8; + w |= (uint32_t) src[1] << 16; + w |= (uint32_t) src[0] << 24; + return w; +#endif +} + +#define STORE32_BE(DST, W) store32_be((DST), (W)) +static inline void +store32_be(uint8_t dst[4], uint32_t w) +{ +#ifdef NATIVE_BIG_ENDIAN + memcpy(dst, &w, sizeof w); +#else + dst[3] = (uint8_t) w; w >>= 8; + dst[2] = (uint8_t) w; w >>= 8; + dst[1] = (uint8_t) w; w >>= 8; + dst[0] = (uint8_t) w; +#endif +} + +#define XOR_BUF(OUT, IN, N) xor_buf((OUT), (IN), (N)) +static inline void +xor_buf(unsigned char *out, const unsigned char *in, size_t n) +{ + size_t i; + + for (i = 0; i < n; i++) { + out[i] ^= in[i]; + } +} + + +#if !defined(__clang__) && !defined(__GNUC__) +# ifdef __attribute__ +# undef __attribute__ +# endif +# define __attribute__(a) +#endif + +#ifndef CRYPTO_ALIGN +# if defined(__INTEL_COMPILER) || defined(_MSC_VER) +# define CRYPTO_ALIGN(x) __declspec(align(x)) +# else +# define CRYPTO_ALIGN(x) __attribute__ ((aligned(x))) +# endif +#endif + +#if defined(__clang__) || defined(__GNUC__) + + #if NETCODE_AVX2 + + # define HAVE_MMINTRIN_H 1 + # define HAVE_EMMINTRIN_H 1 + # define HAVE_PMMINTRIN_H 1 + # define HAVE_TMMINTRIN_H 1 + # define HAVE_SMMINTRIN_H 1 + # define HAVE_AVXINTRIN_H 1 + # define HAVE_WMMINTRIN_H 1 + # define HAVE_AVX2INTRIN_H 1 + # define HAVE_AVX_ASM 1 + # define HAVE_AMD64_ASM 1 + # define HAVE_CPUID 1 + + #elif NETCODE_AVX + + # define HAVE_MMINTRIN_H 1 + # define HAVE_EMMINTRIN_H 1 + # define HAVE_PMMINTRIN_H 1 + # define HAVE_TMMINTRIN_H 1 + # define HAVE_SMMINTRIN_H 1 + # define HAVE_AVXINTRIN_H 1 + # define HAVE_WMMINTRIN_H 1 + # define HAVE_AVX_ASM 1 + # define HAVE_AMD64_ASM 1 + # define HAVE_CPUID 1 + + #elif NETCODE_X64 + + # define HAVE_MMINTRIN_H 1 + # define HAVE_EMMINTRIN_H 1 + # define HAVE_PMMINTRIN_H 1 + # define HAVE_TMMINTRIN_H 1 + # define HAVE_SMMINTRIN_H 1 + # define HAVE_WMMINTRIN_H 1 + # define HAVE_AMD64_ASM 1 + # define HAVE_CPUID 1 + + #elif NETCODE_X64 + + # define HAVE_MMINTRIN_H 1 + # define HAVE_EMMINTRIN_H 1 + # define HAVE_PMMINTRIN_H 1 + # define HAVE_TMMINTRIN_H 1 + # define HAVE_SMMINTRIN_H 1 + # define HAVE_WMMINTRIN_H 1 + # define HAVE_CPUID 1 + + #endif + +#endif + +#if defined(_MSC_VER) && (defined(_M_X64) || defined(_M_AMD64) || defined(_M_IX86)) + +# include + +# define HAVE_INTRIN_H 1 +# define HAVE_MMINTRIN_H 1 +# define HAVE_EMMINTRIN_H 1 +# define HAVE_PMMINTRIN_H 1 +# define HAVE_TMMINTRIN_H 1 +# define HAVE_SMMINTRIN_H 1 + +#if NETCODE_AVX +# define HAVE_AVXINTRIN_H 1 +#endif + +# if _MSC_VER >= 1600 +# define HAVE_WMMINTRIN_H 1 +# endif + +#if NETCODE_AVX2 +# if _MSC_VER >= 1700 && defined(_M_X64) +# define HAVE_AVX2INTRIN_H 1 +# endif +#endif + +#elif defined(HAVE_INTRIN_H) + +# include + +#endif + +#ifdef HAVE_LIBCTGRIND +extern void ct_poison (const void *, size_t); +extern void ct_unpoison(const void *, size_t); +# define POISON(X, L) ct_poison((X), (L)) +# define UNPOISON(X, L) ct_unpoison((X), (L)) +#else +# define POISON(X, L) (void) 0 +# define UNPOISON(X, L) (void) 0 +#endif + +#endif diff --git a/sodium/sodium_private_ed25519_ref10.h b/sodium/sodium_private_ed25519_ref10.h new file mode 100644 index 00000000..b351ae3c --- /dev/null +++ b/sodium/sodium_private_ed25519_ref10.h @@ -0,0 +1,127 @@ +#ifndef ed25519_ref10_H +#define ed25519_ref10_H + +#include +#include + +/* + fe means field element. + Here the field is \Z/(2^255-19). + */ + +#ifdef HAVE_TI_MODE +typedef uint64_t fe25519[5]; +#else +typedef int32_t fe25519[10]; +#endif + +void fe25519_invert(fe25519 out, const fe25519 z); +void fe25519_frombytes(fe25519 h, const unsigned char *s); +void fe25519_tobytes(unsigned char *s, const fe25519 h); + +#ifdef HAVE_TI_MODE +# include "sodium_private_ed25519_ref10_fe_51.h" +#else +# include "sodium_private_ed25519_ref10_fe_25_5.h" +#endif + + +/* + ge means group element. + + Here the group is the set of pairs (x,y) of field elements + satisfying -x^2 + y^2 = 1 + d x^2y^2 + where d = -121665/121666. + + Representations: + ge25519_p2 (projective): (X:Y:Z) satisfying x=X/Z, y=Y/Z + ge25519_p3 (extended): (X:Y:Z:T) satisfying x=X/Z, y=Y/Z, XY=ZT + ge25519_p1p1 (completed): ((X:Z),(Y:T)) satisfying x=X/Z, y=Y/T + ge25519_precomp (Duif): (y+x,y-x,2dxy) + */ + +typedef struct { + fe25519 X; + fe25519 Y; + fe25519 Z; +} ge25519_p2; + +typedef struct { + fe25519 X; + fe25519 Y; + fe25519 Z; + fe25519 T; +} ge25519_p3; + +typedef struct { + fe25519 X; + fe25519 Y; + fe25519 Z; + fe25519 T; +} ge25519_p1p1; + +typedef struct { + fe25519 yplusx; + fe25519 yminusx; + fe25519 xy2d; +} ge25519_precomp; + +typedef struct { + fe25519 YplusX; + fe25519 YminusX; + fe25519 Z; + fe25519 T2d; +} ge25519_cached; + +void ge25519_tobytes(unsigned char *s, const ge25519_p2 *h); + +void ge25519_p3_tobytes(unsigned char *s, const ge25519_p3 *h); + +int ge25519_frombytes(ge25519_p3 *h, const unsigned char *s); + +int ge25519_frombytes_negate_vartime(ge25519_p3 *h, const unsigned char *s); + +void ge25519_p3_to_cached(ge25519_cached *r, const ge25519_p3 *p); + +void ge25519_p1p1_to_p2(ge25519_p2 *r, const ge25519_p1p1 *p); + +void ge25519_p1p1_to_p3(ge25519_p3 *r, const ge25519_p1p1 *p); + +void ge25519_add(ge25519_p1p1 *r, const ge25519_p3 *p, const ge25519_cached *q); + +void ge25519_sub(ge25519_p1p1 *r, const ge25519_p3 *p, const ge25519_cached *q); + +void ge25519_scalarmult_base(ge25519_p3 *h, const unsigned char *a); + +void ge25519_double_scalarmult_vartime(ge25519_p2 *r, const unsigned char *a, + const ge25519_p3 *A, + const unsigned char *b); + +void ge25519_scalarmult(ge25519_p3 *h, const unsigned char *a, + const ge25519_p3 *p); + +int ge25519_is_canonical(const unsigned char *s); + +int ge25519_is_on_curve(const ge25519_p3 *p); + +int ge25519_is_on_main_subgroup(const ge25519_p3 *p); + +int ge25519_has_small_order(const unsigned char s[32]); + +void ge25519_from_uniform(unsigned char s[32], const unsigned char r[32]); + +/* + The set of scalars is \Z/l + where l = 2^252 + 27742317777372353535851937790883648493. + */ + +void sc25519_invert(unsigned char recip[32], const unsigned char s[32]); + +void sc25519_reduce(unsigned char s[64]); + +void sc25519_muladd(unsigned char s[32], const unsigned char a[32], + const unsigned char b[32], const unsigned char c[32]); + +int sc25519_is_canonical(const unsigned char s[32]); + +#endif diff --git a/sodium/sodium_private_ed25519_ref10_fe_25_5.h b/sodium/sodium_private_ed25519_ref10_fe_25_5.h new file mode 100644 index 00000000..94443055 --- /dev/null +++ b/sodium/sodium_private_ed25519_ref10_fe_25_5.h @@ -0,0 +1,1060 @@ +#include + +#include "sodium_private_common.h" +#include "sodium_utils.h" + +/* + h = 0 + */ + +static inline void +fe25519_0(fe25519 h) +{ + memset(&h[0], 0, 10 * sizeof h[0]); +} + +/* + h = 1 + */ + +static inline void +fe25519_1(fe25519 h) +{ + h[0] = 1; + h[1] = 0; + memset(&h[2], 0, 8 * sizeof h[0]); +} + +/* + h = f + g + Can overlap h with f or g. + * + Preconditions: + |f| bounded by 1.1*2^25,1.1*2^24,1.1*2^25,1.1*2^24,etc. + |g| bounded by 1.1*2^25,1.1*2^24,1.1*2^25,1.1*2^24,etc. + * + Postconditions: + |h| bounded by 1.1*2^26,1.1*2^25,1.1*2^26,1.1*2^25,etc. + */ + +static inline void +fe25519_add(fe25519 h, const fe25519 f, const fe25519 g) +{ + int32_t h0 = f[0] + g[0]; + int32_t h1 = f[1] + g[1]; + int32_t h2 = f[2] + g[2]; + int32_t h3 = f[3] + g[3]; + int32_t h4 = f[4] + g[4]; + int32_t h5 = f[5] + g[5]; + int32_t h6 = f[6] + g[6]; + int32_t h7 = f[7] + g[7]; + int32_t h8 = f[8] + g[8]; + int32_t h9 = f[9] + g[9]; + + h[0] = h0; + h[1] = h1; + h[2] = h2; + h[3] = h3; + h[4] = h4; + h[5] = h5; + h[6] = h6; + h[7] = h7; + h[8] = h8; + h[9] = h9; +} + +#if defined(__ORBIS__) || defined(__PROSPERO__) || defined(NN_NINTENDO_SDK) +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wunused-function" +#endif // #if defined(__ORBIS__) || defined(__PROSPERO__) || defined(NN_NINTENDO_SDK) + +/* + h = f - g + Can overlap h with f or g. + * + Preconditions: + |f| bounded by 1.1*2^25,1.1*2^24,1.1*2^25,1.1*2^24,etc. + |g| bounded by 1.1*2^25,1.1*2^24,1.1*2^25,1.1*2^24,etc. + * + Postconditions: + |h| bounded by 1.1*2^26,1.1*2^25,1.1*2^26,1.1*2^25,etc. + */ + +static void +fe25519_sub(fe25519 h, const fe25519 f, const fe25519 g) +{ + int32_t h0 = f[0] - g[0]; + int32_t h1 = f[1] - g[1]; + int32_t h2 = f[2] - g[2]; + int32_t h3 = f[3] - g[3]; + int32_t h4 = f[4] - g[4]; + int32_t h5 = f[5] - g[5]; + int32_t h6 = f[6] - g[6]; + int32_t h7 = f[7] - g[7]; + int32_t h8 = f[8] - g[8]; + int32_t h9 = f[9] - g[9]; + + h[0] = h0; + h[1] = h1; + h[2] = h2; + h[3] = h3; + h[4] = h4; + h[5] = h5; + h[6] = h6; + h[7] = h7; + h[8] = h8; + h[9] = h9; +} + +/* + h = -f + * + Preconditions: + |f| bounded by 1.1*2^25,1.1*2^24,1.1*2^25,1.1*2^24,etc. + * + Postconditions: + |h| bounded by 1.1*2^25,1.1*2^24,1.1*2^25,1.1*2^24,etc. + */ + +static inline void +fe25519_neg(fe25519 h, const fe25519 f) +{ + int32_t h0 = -f[0]; + int32_t h1 = -f[1]; + int32_t h2 = -f[2]; + int32_t h3 = -f[3]; + int32_t h4 = -f[4]; + int32_t h5 = -f[5]; + int32_t h6 = -f[6]; + int32_t h7 = -f[7]; + int32_t h8 = -f[8]; + int32_t h9 = -f[9]; + + h[0] = h0; + h[1] = h1; + h[2] = h2; + h[3] = h3; + h[4] = h4; + h[5] = h5; + h[6] = h6; + h[7] = h7; + h[8] = h8; + h[9] = h9; +} + +/* + Replace (f,g) with (g,g) if b == 1; + replace (f,g) with (f,g) if b == 0. + * + Preconditions: b in {0,1}. + */ + +static void +fe25519_cmov(fe25519 f, const fe25519 g, unsigned int b) +{ + const uint32_t mask = (uint32_t) (-(int32_t) b); + + int32_t f0 = f[0]; + int32_t f1 = f[1]; + int32_t f2 = f[2]; + int32_t f3 = f[3]; + int32_t f4 = f[4]; + int32_t f5 = f[5]; + int32_t f6 = f[6]; + int32_t f7 = f[7]; + int32_t f8 = f[8]; + int32_t f9 = f[9]; + + int32_t x0 = f0 ^ g[0]; + int32_t x1 = f1 ^ g[1]; + int32_t x2 = f2 ^ g[2]; + int32_t x3 = f3 ^ g[3]; + int32_t x4 = f4 ^ g[4]; + int32_t x5 = f5 ^ g[5]; + int32_t x6 = f6 ^ g[6]; + int32_t x7 = f7 ^ g[7]; + int32_t x8 = f8 ^ g[8]; + int32_t x9 = f9 ^ g[9]; + + x0 &= mask; + x1 &= mask; + x2 &= mask; + x3 &= mask; + x4 &= mask; + x5 &= mask; + x6 &= mask; + x7 &= mask; + x8 &= mask; + x9 &= mask; + + f[0] = f0 ^ x0; + f[1] = f1 ^ x1; + f[2] = f2 ^ x2; + f[3] = f3 ^ x3; + f[4] = f4 ^ x4; + f[5] = f5 ^ x5; + f[6] = f6 ^ x6; + f[7] = f7 ^ x7; + f[8] = f8 ^ x8; + f[9] = f9 ^ x9; +} + +static void +fe25519_cswap(fe25519 f, fe25519 g, unsigned int b) +{ + const uint32_t mask = (uint32_t) (-(int64_t) b); + + int32_t f0 = f[0]; + int32_t f1 = f[1]; + int32_t f2 = f[2]; + int32_t f3 = f[3]; + int32_t f4 = f[4]; + int32_t f5 = f[5]; + int32_t f6 = f[6]; + int32_t f7 = f[7]; + int32_t f8 = f[8]; + int32_t f9 = f[9]; + + int32_t g0 = g[0]; + int32_t g1 = g[1]; + int32_t g2 = g[2]; + int32_t g3 = g[3]; + int32_t g4 = g[4]; + int32_t g5 = g[5]; + int32_t g6 = g[6]; + int32_t g7 = g[7]; + int32_t g8 = g[8]; + int32_t g9 = g[9]; + + int32_t x0 = f0 ^ g0; + int32_t x1 = f1 ^ g1; + int32_t x2 = f2 ^ g2; + int32_t x3 = f3 ^ g3; + int32_t x4 = f4 ^ g4; + int32_t x5 = f5 ^ g5; + int32_t x6 = f6 ^ g6; + int32_t x7 = f7 ^ g7; + int32_t x8 = f8 ^ g8; + int32_t x9 = f9 ^ g9; + + x0 &= mask; + x1 &= mask; + x2 &= mask; + x3 &= mask; + x4 &= mask; + x5 &= mask; + x6 &= mask; + x7 &= mask; + x8 &= mask; + x9 &= mask; + + f[0] = f0 ^ x0; + f[1] = f1 ^ x1; + f[2] = f2 ^ x2; + f[3] = f3 ^ x3; + f[4] = f4 ^ x4; + f[5] = f5 ^ x5; + f[6] = f6 ^ x6; + f[7] = f7 ^ x7; + f[8] = f8 ^ x8; + f[9] = f9 ^ x9; + + g[0] = g0 ^ x0; + g[1] = g1 ^ x1; + g[2] = g2 ^ x2; + g[3] = g3 ^ x3; + g[4] = g4 ^ x4; + g[5] = g5 ^ x5; + g[6] = g6 ^ x6; + g[7] = g7 ^ x7; + g[8] = g8 ^ x8; + g[9] = g9 ^ x9; +} + +/* + h = f + */ + +static inline void +fe25519_copy(fe25519 h, const fe25519 f) +{ + int32_t f0 = f[0]; + int32_t f1 = f[1]; + int32_t f2 = f[2]; + int32_t f3 = f[3]; + int32_t f4 = f[4]; + int32_t f5 = f[5]; + int32_t f6 = f[6]; + int32_t f7 = f[7]; + int32_t f8 = f[8]; + int32_t f9 = f[9]; + + h[0] = f0; + h[1] = f1; + h[2] = f2; + h[3] = f3; + h[4] = f4; + h[5] = f5; + h[6] = f6; + h[7] = f7; + h[8] = f8; + h[9] = f9; +} + +/* + return 1 if f is in {1,3,5,...,q-2} + return 0 if f is in {0,2,4,...,q-1} + + Preconditions: + |f| bounded by 1.1*2^26,1.1*2^25,1.1*2^26,1.1*2^25,etc. + */ + +static inline int +fe25519_isnegative(const fe25519 f) +{ + unsigned char s[32]; + + fe25519_tobytes(s, f); + + return s[0] & 1; +} + +/* + return 1 if f == 0 + return 0 if f != 0 + + Preconditions: + |f| bounded by 1.1*2^26,1.1*2^25,1.1*2^26,1.1*2^25,etc. + */ + +static inline int +fe25519_iszero(const fe25519 f) +{ + unsigned char s[32]; + + fe25519_tobytes(s, f); + + return sodium_is_zero(s, 32); +} + +/* + h = f * g + Can overlap h with f or g. + * + Preconditions: + |f| bounded by 1.65*2^26,1.65*2^25,1.65*2^26,1.65*2^25,etc. + |g| bounded by 1.65*2^26,1.65*2^25,1.65*2^26,1.65*2^25,etc. + * + Postconditions: + |h| bounded by 1.01*2^25,1.01*2^24,1.01*2^25,1.01*2^24,etc. + */ + +/* + Notes on implementation strategy: + * + Using schoolbook multiplication. + Karatsuba would save a little in some cost models. + * + Most multiplications by 2 and 19 are 32-bit precomputations; + cheaper than 64-bit postcomputations. + * + There is one remaining multiplication by 19 in the carry chain; + one *19 precomputation can be merged into this, + but the resulting data flow is considerably less clean. + * + There are 12 carries below. + 10 of them are 2-way parallelizable and vectorizable. + Can get away with 11 carries, but then data flow is much deeper. + * + With tighter constraints on inputs can squeeze carries into int32. + */ + +static void +fe25519_mul(fe25519 h, const fe25519 f, const fe25519 g) +{ + int32_t f0 = f[0]; + int32_t f1 = f[1]; + int32_t f2 = f[2]; + int32_t f3 = f[3]; + int32_t f4 = f[4]; + int32_t f5 = f[5]; + int32_t f6 = f[6]; + int32_t f7 = f[7]; + int32_t f8 = f[8]; + int32_t f9 = f[9]; + + int32_t g0 = g[0]; + int32_t g1 = g[1]; + int32_t g2 = g[2]; + int32_t g3 = g[3]; + int32_t g4 = g[4]; + int32_t g5 = g[5]; + int32_t g6 = g[6]; + int32_t g7 = g[7]; + int32_t g8 = g[8]; + int32_t g9 = g[9]; + + int32_t g1_19 = 19 * g1; /* 1.959375*2^29 */ + int32_t g2_19 = 19 * g2; /* 1.959375*2^30; still ok */ + int32_t g3_19 = 19 * g3; + int32_t g4_19 = 19 * g4; + int32_t g5_19 = 19 * g5; + int32_t g6_19 = 19 * g6; + int32_t g7_19 = 19 * g7; + int32_t g8_19 = 19 * g8; + int32_t g9_19 = 19 * g9; + int32_t f1_2 = 2 * f1; + int32_t f3_2 = 2 * f3; + int32_t f5_2 = 2 * f5; + int32_t f7_2 = 2 * f7; + int32_t f9_2 = 2 * f9; + + int64_t f0g0 = f0 * (int64_t) g0; + int64_t f0g1 = f0 * (int64_t) g1; + int64_t f0g2 = f0 * (int64_t) g2; + int64_t f0g3 = f0 * (int64_t) g3; + int64_t f0g4 = f0 * (int64_t) g4; + int64_t f0g5 = f0 * (int64_t) g5; + int64_t f0g6 = f0 * (int64_t) g6; + int64_t f0g7 = f0 * (int64_t) g7; + int64_t f0g8 = f0 * (int64_t) g8; + int64_t f0g9 = f0 * (int64_t) g9; + int64_t f1g0 = f1 * (int64_t) g0; + int64_t f1g1_2 = f1_2 * (int64_t) g1; + int64_t f1g2 = f1 * (int64_t) g2; + int64_t f1g3_2 = f1_2 * (int64_t) g3; + int64_t f1g4 = f1 * (int64_t) g4; + int64_t f1g5_2 = f1_2 * (int64_t) g5; + int64_t f1g6 = f1 * (int64_t) g6; + int64_t f1g7_2 = f1_2 * (int64_t) g7; + int64_t f1g8 = f1 * (int64_t) g8; + int64_t f1g9_38 = f1_2 * (int64_t) g9_19; + int64_t f2g0 = f2 * (int64_t) g0; + int64_t f2g1 = f2 * (int64_t) g1; + int64_t f2g2 = f2 * (int64_t) g2; + int64_t f2g3 = f2 * (int64_t) g3; + int64_t f2g4 = f2 * (int64_t) g4; + int64_t f2g5 = f2 * (int64_t) g5; + int64_t f2g6 = f2 * (int64_t) g6; + int64_t f2g7 = f2 * (int64_t) g7; + int64_t f2g8_19 = f2 * (int64_t) g8_19; + int64_t f2g9_19 = f2 * (int64_t) g9_19; + int64_t f3g0 = f3 * (int64_t) g0; + int64_t f3g1_2 = f3_2 * (int64_t) g1; + int64_t f3g2 = f3 * (int64_t) g2; + int64_t f3g3_2 = f3_2 * (int64_t) g3; + int64_t f3g4 = f3 * (int64_t) g4; + int64_t f3g5_2 = f3_2 * (int64_t) g5; + int64_t f3g6 = f3 * (int64_t) g6; + int64_t f3g7_38 = f3_2 * (int64_t) g7_19; + int64_t f3g8_19 = f3 * (int64_t) g8_19; + int64_t f3g9_38 = f3_2 * (int64_t) g9_19; + int64_t f4g0 = f4 * (int64_t) g0; + int64_t f4g1 = f4 * (int64_t) g1; + int64_t f4g2 = f4 * (int64_t) g2; + int64_t f4g3 = f4 * (int64_t) g3; + int64_t f4g4 = f4 * (int64_t) g4; + int64_t f4g5 = f4 * (int64_t) g5; + int64_t f4g6_19 = f4 * (int64_t) g6_19; + int64_t f4g7_19 = f4 * (int64_t) g7_19; + int64_t f4g8_19 = f4 * (int64_t) g8_19; + int64_t f4g9_19 = f4 * (int64_t) g9_19; + int64_t f5g0 = f5 * (int64_t) g0; + int64_t f5g1_2 = f5_2 * (int64_t) g1; + int64_t f5g2 = f5 * (int64_t) g2; + int64_t f5g3_2 = f5_2 * (int64_t) g3; + int64_t f5g4 = f5 * (int64_t) g4; + int64_t f5g5_38 = f5_2 * (int64_t) g5_19; + int64_t f5g6_19 = f5 * (int64_t) g6_19; + int64_t f5g7_38 = f5_2 * (int64_t) g7_19; + int64_t f5g8_19 = f5 * (int64_t) g8_19; + int64_t f5g9_38 = f5_2 * (int64_t) g9_19; + int64_t f6g0 = f6 * (int64_t) g0; + int64_t f6g1 = f6 * (int64_t) g1; + int64_t f6g2 = f6 * (int64_t) g2; + int64_t f6g3 = f6 * (int64_t) g3; + int64_t f6g4_19 = f6 * (int64_t) g4_19; + int64_t f6g5_19 = f6 * (int64_t) g5_19; + int64_t f6g6_19 = f6 * (int64_t) g6_19; + int64_t f6g7_19 = f6 * (int64_t) g7_19; + int64_t f6g8_19 = f6 * (int64_t) g8_19; + int64_t f6g9_19 = f6 * (int64_t) g9_19; + int64_t f7g0 = f7 * (int64_t) g0; + int64_t f7g1_2 = f7_2 * (int64_t) g1; + int64_t f7g2 = f7 * (int64_t) g2; + int64_t f7g3_38 = f7_2 * (int64_t) g3_19; + int64_t f7g4_19 = f7 * (int64_t) g4_19; + int64_t f7g5_38 = f7_2 * (int64_t) g5_19; + int64_t f7g6_19 = f7 * (int64_t) g6_19; + int64_t f7g7_38 = f7_2 * (int64_t) g7_19; + int64_t f7g8_19 = f7 * (int64_t) g8_19; + int64_t f7g9_38 = f7_2 * (int64_t) g9_19; + int64_t f8g0 = f8 * (int64_t) g0; + int64_t f8g1 = f8 * (int64_t) g1; + int64_t f8g2_19 = f8 * (int64_t) g2_19; + int64_t f8g3_19 = f8 * (int64_t) g3_19; + int64_t f8g4_19 = f8 * (int64_t) g4_19; + int64_t f8g5_19 = f8 * (int64_t) g5_19; + int64_t f8g6_19 = f8 * (int64_t) g6_19; + int64_t f8g7_19 = f8 * (int64_t) g7_19; + int64_t f8g8_19 = f8 * (int64_t) g8_19; + int64_t f8g9_19 = f8 * (int64_t) g9_19; + int64_t f9g0 = f9 * (int64_t) g0; + int64_t f9g1_38 = f9_2 * (int64_t) g1_19; + int64_t f9g2_19 = f9 * (int64_t) g2_19; + int64_t f9g3_38 = f9_2 * (int64_t) g3_19; + int64_t f9g4_19 = f9 * (int64_t) g4_19; + int64_t f9g5_38 = f9_2 * (int64_t) g5_19; + int64_t f9g6_19 = f9 * (int64_t) g6_19; + int64_t f9g7_38 = f9_2 * (int64_t) g7_19; + int64_t f9g8_19 = f9 * (int64_t) g8_19; + int64_t f9g9_38 = f9_2 * (int64_t) g9_19; + + int64_t h0 = f0g0 + f1g9_38 + f2g8_19 + f3g7_38 + f4g6_19 + f5g5_38 + + f6g4_19 + f7g3_38 + f8g2_19 + f9g1_38; + int64_t h1 = f0g1 + f1g0 + f2g9_19 + f3g8_19 + f4g7_19 + f5g6_19 + f6g5_19 + + f7g4_19 + f8g3_19 + f9g2_19; + int64_t h2 = f0g2 + f1g1_2 + f2g0 + f3g9_38 + f4g8_19 + f5g7_38 + f6g6_19 + + f7g5_38 + f8g4_19 + f9g3_38; + int64_t h3 = f0g3 + f1g2 + f2g1 + f3g0 + f4g9_19 + f5g8_19 + f6g7_19 + + f7g6_19 + f8g5_19 + f9g4_19; + int64_t h4 = f0g4 + f1g3_2 + f2g2 + f3g1_2 + f4g0 + f5g9_38 + f6g8_19 + + f7g7_38 + f8g6_19 + f9g5_38; + int64_t h5 = f0g5 + f1g4 + f2g3 + f3g2 + f4g1 + f5g0 + f6g9_19 + f7g8_19 + + f8g7_19 + f9g6_19; + int64_t h6 = f0g6 + f1g5_2 + f2g4 + f3g3_2 + f4g2 + f5g1_2 + f6g0 + + f7g9_38 + f8g8_19 + f9g7_38; + int64_t h7 = f0g7 + f1g6 + f2g5 + f3g4 + f4g3 + f5g2 + f6g1 + f7g0 + + f8g9_19 + f9g8_19; + int64_t h8 = f0g8 + f1g7_2 + f2g6 + f3g5_2 + f4g4 + f5g3_2 + f6g2 + f7g1_2 + + f8g0 + f9g9_38; + int64_t h9 = + f0g9 + f1g8 + f2g7 + f3g6 + f4g5 + f5g4 + f6g3 + f7g2 + f8g1 + f9g0; + + int64_t carry0; + int64_t carry1; + int64_t carry2; + int64_t carry3; + int64_t carry4; + int64_t carry5; + int64_t carry6; + int64_t carry7; + int64_t carry8; + int64_t carry9; + + /* + |h0| <= (1.65*1.65*2^52*(1+19+19+19+19)+1.65*1.65*2^50*(38+38+38+38+38)) + i.e. |h0| <= 1.4*2^60; narrower ranges for h2, h4, h6, h8 + |h1| <= (1.65*1.65*2^51*(1+1+19+19+19+19+19+19+19+19)) + i.e. |h1| <= 1.7*2^59; narrower ranges for h3, h5, h7, h9 + */ + + carry0 = (h0 + (int64_t)(1L << 25)) >> 26; + h1 += carry0; + h0 -= carry0 * ((uint64_t) 1L << 26); + carry4 = (h4 + (int64_t)(1L << 25)) >> 26; + h5 += carry4; + h4 -= carry4 * ((uint64_t) 1L << 26); + /* |h0| <= 2^25 */ + /* |h4| <= 2^25 */ + /* |h1| <= 1.71*2^59 */ + /* |h5| <= 1.71*2^59 */ + + carry1 = (h1 + (int64_t)(1L << 24)) >> 25; + h2 += carry1; + h1 -= carry1 * ((uint64_t) 1L << 25); + carry5 = (h5 + (int64_t)(1L << 24)) >> 25; + h6 += carry5; + h5 -= carry5 * ((uint64_t) 1L << 25); + /* |h1| <= 2^24; from now on fits into int32 */ + /* |h5| <= 2^24; from now on fits into int32 */ + /* |h2| <= 1.41*2^60 */ + /* |h6| <= 1.41*2^60 */ + + carry2 = (h2 + (int64_t)(1L << 25)) >> 26; + h3 += carry2; + h2 -= carry2 * ((uint64_t) 1L << 26); + carry6 = (h6 + (int64_t)(1L << 25)) >> 26; + h7 += carry6; + h6 -= carry6 * ((uint64_t) 1L << 26); + /* |h2| <= 2^25; from now on fits into int32 unchanged */ + /* |h6| <= 2^25; from now on fits into int32 unchanged */ + /* |h3| <= 1.71*2^59 */ + /* |h7| <= 1.71*2^59 */ + + carry3 = (h3 + (int64_t)(1L << 24)) >> 25; + h4 += carry3; + h3 -= carry3 * ((uint64_t) 1L << 25); + carry7 = (h7 + (int64_t)(1L << 24)) >> 25; + h8 += carry7; + h7 -= carry7 * ((uint64_t) 1L << 25); + /* |h3| <= 2^24; from now on fits into int32 unchanged */ + /* |h7| <= 2^24; from now on fits into int32 unchanged */ + /* |h4| <= 1.72*2^34 */ + /* |h8| <= 1.41*2^60 */ + + carry4 = (h4 + (int64_t)(1L << 25)) >> 26; + h5 += carry4; + h4 -= carry4 * ((uint64_t) 1L << 26); + carry8 = (h8 + (int64_t)(1L << 25)) >> 26; + h9 += carry8; + h8 -= carry8 * ((uint64_t) 1L << 26); + /* |h4| <= 2^25; from now on fits into int32 unchanged */ + /* |h8| <= 2^25; from now on fits into int32 unchanged */ + /* |h5| <= 1.01*2^24 */ + /* |h9| <= 1.71*2^59 */ + + carry9 = (h9 + (int64_t)(1L << 24)) >> 25; + h0 += carry9 * 19; + h9 -= carry9 * ((uint64_t) 1L << 25); + /* |h9| <= 2^24; from now on fits into int32 unchanged */ + /* |h0| <= 1.1*2^39 */ + + carry0 = (h0 + (int64_t)(1L << 25)) >> 26; + h1 += carry0; + h0 -= carry0 * ((uint64_t) 1L << 26); + /* |h0| <= 2^25; from now on fits into int32 unchanged */ + /* |h1| <= 1.01*2^24 */ + + h[0] = (int32_t) h0; + h[1] = (int32_t) h1; + h[2] = (int32_t) h2; + h[3] = (int32_t) h3; + h[4] = (int32_t) h4; + h[5] = (int32_t) h5; + h[6] = (int32_t) h6; + h[7] = (int32_t) h7; + h[8] = (int32_t) h8; + h[9] = (int32_t) h9; +} + +/* + h = f * f + Can overlap h with f. + * + Preconditions: + |f| bounded by 1.65*2^26,1.65*2^25,1.65*2^26,1.65*2^25,etc. + * + Postconditions: + |h| bounded by 1.01*2^25,1.01*2^24,1.01*2^25,1.01*2^24,etc. + */ + +static void +fe25519_sq(fe25519 h, const fe25519 f) +{ + int32_t f0 = f[0]; + int32_t f1 = f[1]; + int32_t f2 = f[2]; + int32_t f3 = f[3]; + int32_t f4 = f[4]; + int32_t f5 = f[5]; + int32_t f6 = f[6]; + int32_t f7 = f[7]; + int32_t f8 = f[8]; + int32_t f9 = f[9]; + + int32_t f0_2 = 2 * f0; + int32_t f1_2 = 2 * f1; + int32_t f2_2 = 2 * f2; + int32_t f3_2 = 2 * f3; + int32_t f4_2 = 2 * f4; + int32_t f5_2 = 2 * f5; + int32_t f6_2 = 2 * f6; + int32_t f7_2 = 2 * f7; + int32_t f5_38 = 38 * f5; /* 1.959375*2^30 */ + int32_t f6_19 = 19 * f6; /* 1.959375*2^30 */ + int32_t f7_38 = 38 * f7; /* 1.959375*2^30 */ + int32_t f8_19 = 19 * f8; /* 1.959375*2^30 */ + int32_t f9_38 = 38 * f9; /* 1.959375*2^30 */ + + int64_t f0f0 = f0 * (int64_t) f0; + int64_t f0f1_2 = f0_2 * (int64_t) f1; + int64_t f0f2_2 = f0_2 * (int64_t) f2; + int64_t f0f3_2 = f0_2 * (int64_t) f3; + int64_t f0f4_2 = f0_2 * (int64_t) f4; + int64_t f0f5_2 = f0_2 * (int64_t) f5; + int64_t f0f6_2 = f0_2 * (int64_t) f6; + int64_t f0f7_2 = f0_2 * (int64_t) f7; + int64_t f0f8_2 = f0_2 * (int64_t) f8; + int64_t f0f9_2 = f0_2 * (int64_t) f9; + int64_t f1f1_2 = f1_2 * (int64_t) f1; + int64_t f1f2_2 = f1_2 * (int64_t) f2; + int64_t f1f3_4 = f1_2 * (int64_t) f3_2; + int64_t f1f4_2 = f1_2 * (int64_t) f4; + int64_t f1f5_4 = f1_2 * (int64_t) f5_2; + int64_t f1f6_2 = f1_2 * (int64_t) f6; + int64_t f1f7_4 = f1_2 * (int64_t) f7_2; + int64_t f1f8_2 = f1_2 * (int64_t) f8; + int64_t f1f9_76 = f1_2 * (int64_t) f9_38; + int64_t f2f2 = f2 * (int64_t) f2; + int64_t f2f3_2 = f2_2 * (int64_t) f3; + int64_t f2f4_2 = f2_2 * (int64_t) f4; + int64_t f2f5_2 = f2_2 * (int64_t) f5; + int64_t f2f6_2 = f2_2 * (int64_t) f6; + int64_t f2f7_2 = f2_2 * (int64_t) f7; + int64_t f2f8_38 = f2_2 * (int64_t) f8_19; + int64_t f2f9_38 = f2 * (int64_t) f9_38; + int64_t f3f3_2 = f3_2 * (int64_t) f3; + int64_t f3f4_2 = f3_2 * (int64_t) f4; + int64_t f3f5_4 = f3_2 * (int64_t) f5_2; + int64_t f3f6_2 = f3_2 * (int64_t) f6; + int64_t f3f7_76 = f3_2 * (int64_t) f7_38; + int64_t f3f8_38 = f3_2 * (int64_t) f8_19; + int64_t f3f9_76 = f3_2 * (int64_t) f9_38; + int64_t f4f4 = f4 * (int64_t) f4; + int64_t f4f5_2 = f4_2 * (int64_t) f5; + int64_t f4f6_38 = f4_2 * (int64_t) f6_19; + int64_t f4f7_38 = f4 * (int64_t) f7_38; + int64_t f4f8_38 = f4_2 * (int64_t) f8_19; + int64_t f4f9_38 = f4 * (int64_t) f9_38; + int64_t f5f5_38 = f5 * (int64_t) f5_38; + int64_t f5f6_38 = f5_2 * (int64_t) f6_19; + int64_t f5f7_76 = f5_2 * (int64_t) f7_38; + int64_t f5f8_38 = f5_2 * (int64_t) f8_19; + int64_t f5f9_76 = f5_2 * (int64_t) f9_38; + int64_t f6f6_19 = f6 * (int64_t) f6_19; + int64_t f6f7_38 = f6 * (int64_t) f7_38; + int64_t f6f8_38 = f6_2 * (int64_t) f8_19; + int64_t f6f9_38 = f6 * (int64_t) f9_38; + int64_t f7f7_38 = f7 * (int64_t) f7_38; + int64_t f7f8_38 = f7_2 * (int64_t) f8_19; + int64_t f7f9_76 = f7_2 * (int64_t) f9_38; + int64_t f8f8_19 = f8 * (int64_t) f8_19; + int64_t f8f9_38 = f8 * (int64_t) f9_38; + int64_t f9f9_38 = f9 * (int64_t) f9_38; + + int64_t h0 = f0f0 + f1f9_76 + f2f8_38 + f3f7_76 + f4f6_38 + f5f5_38; + int64_t h1 = f0f1_2 + f2f9_38 + f3f8_38 + f4f7_38 + f5f6_38; + int64_t h2 = f0f2_2 + f1f1_2 + f3f9_76 + f4f8_38 + f5f7_76 + f6f6_19; + int64_t h3 = f0f3_2 + f1f2_2 + f4f9_38 + f5f8_38 + f6f7_38; + int64_t h4 = f0f4_2 + f1f3_4 + f2f2 + f5f9_76 + f6f8_38 + f7f7_38; + int64_t h5 = f0f5_2 + f1f4_2 + f2f3_2 + f6f9_38 + f7f8_38; + int64_t h6 = f0f6_2 + f1f5_4 + f2f4_2 + f3f3_2 + f7f9_76 + f8f8_19; + int64_t h7 = f0f7_2 + f1f6_2 + f2f5_2 + f3f4_2 + f8f9_38; + int64_t h8 = f0f8_2 + f1f7_4 + f2f6_2 + f3f5_4 + f4f4 + f9f9_38; + int64_t h9 = f0f9_2 + f1f8_2 + f2f7_2 + f3f6_2 + f4f5_2; + + int64_t carry0; + int64_t carry1; + int64_t carry2; + int64_t carry3; + int64_t carry4; + int64_t carry5; + int64_t carry6; + int64_t carry7; + int64_t carry8; + int64_t carry9; + + carry0 = (h0 + (int64_t)(1L << 25)) >> 26; + h1 += carry0; + h0 -= carry0 * ((uint64_t) 1L << 26); + carry4 = (h4 + (int64_t)(1L << 25)) >> 26; + h5 += carry4; + h4 -= carry4 * ((uint64_t) 1L << 26); + + carry1 = (h1 + (int64_t)(1L << 24)) >> 25; + h2 += carry1; + h1 -= carry1 * ((uint64_t) 1L << 25); + carry5 = (h5 + (int64_t)(1L << 24)) >> 25; + h6 += carry5; + h5 -= carry5 * ((uint64_t) 1L << 25); + + carry2 = (h2 + (int64_t)(1L << 25)) >> 26; + h3 += carry2; + h2 -= carry2 * ((uint64_t) 1L << 26); + carry6 = (h6 + (int64_t)(1L << 25)) >> 26; + h7 += carry6; + h6 -= carry6 * ((uint64_t) 1L << 26); + + carry3 = (h3 + (int64_t)(1L << 24)) >> 25; + h4 += carry3; + h3 -= carry3 * ((uint64_t) 1L << 25); + carry7 = (h7 + (int64_t)(1L << 24)) >> 25; + h8 += carry7; + h7 -= carry7 * ((uint64_t) 1L << 25); + + carry4 = (h4 + (int64_t)(1L << 25)) >> 26; + h5 += carry4; + h4 -= carry4 * ((uint64_t) 1L << 26); + carry8 = (h8 + (int64_t)(1L << 25)) >> 26; + h9 += carry8; + h8 -= carry8 * ((uint64_t) 1L << 26); + + carry9 = (h9 + (int64_t)(1L << 24)) >> 25; + h0 += carry9 * 19; + h9 -= carry9 * ((uint64_t) 1L << 25); + + carry0 = (h0 + (int64_t)(1L << 25)) >> 26; + h1 += carry0; + h0 -= carry0 * ((uint64_t) 1L << 26); + + h[0] = (int32_t) h0; + h[1] = (int32_t) h1; + h[2] = (int32_t) h2; + h[3] = (int32_t) h3; + h[4] = (int32_t) h4; + h[5] = (int32_t) h5; + h[6] = (int32_t) h6; + h[7] = (int32_t) h7; + h[8] = (int32_t) h8; + h[9] = (int32_t) h9; +} + +/* + h = 2 * f * f + Can overlap h with f. + * + Preconditions: + |f| bounded by 1.65*2^26,1.65*2^25,1.65*2^26,1.65*2^25,etc. + * + Postconditions: + |h| bounded by 1.01*2^25,1.01*2^24,1.01*2^25,1.01*2^24,etc. + */ + +static void +fe25519_sq2(fe25519 h, const fe25519 f) +{ + int32_t f0 = f[0]; + int32_t f1 = f[1]; + int32_t f2 = f[2]; + int32_t f3 = f[3]; + int32_t f4 = f[4]; + int32_t f5 = f[5]; + int32_t f6 = f[6]; + int32_t f7 = f[7]; + int32_t f8 = f[8]; + int32_t f9 = f[9]; + + int32_t f0_2 = 2 * f0; + int32_t f1_2 = 2 * f1; + int32_t f2_2 = 2 * f2; + int32_t f3_2 = 2 * f3; + int32_t f4_2 = 2 * f4; + int32_t f5_2 = 2 * f5; + int32_t f6_2 = 2 * f6; + int32_t f7_2 = 2 * f7; + int32_t f5_38 = 38 * f5; /* 1.959375*2^30 */ + int32_t f6_19 = 19 * f6; /* 1.959375*2^30 */ + int32_t f7_38 = 38 * f7; /* 1.959375*2^30 */ + int32_t f8_19 = 19 * f8; /* 1.959375*2^30 */ + int32_t f9_38 = 38 * f9; /* 1.959375*2^30 */ + + int64_t f0f0 = f0 * (int64_t) f0; + int64_t f0f1_2 = f0_2 * (int64_t) f1; + int64_t f0f2_2 = f0_2 * (int64_t) f2; + int64_t f0f3_2 = f0_2 * (int64_t) f3; + int64_t f0f4_2 = f0_2 * (int64_t) f4; + int64_t f0f5_2 = f0_2 * (int64_t) f5; + int64_t f0f6_2 = f0_2 * (int64_t) f6; + int64_t f0f7_2 = f0_2 * (int64_t) f7; + int64_t f0f8_2 = f0_2 * (int64_t) f8; + int64_t f0f9_2 = f0_2 * (int64_t) f9; + int64_t f1f1_2 = f1_2 * (int64_t) f1; + int64_t f1f2_2 = f1_2 * (int64_t) f2; + int64_t f1f3_4 = f1_2 * (int64_t) f3_2; + int64_t f1f4_2 = f1_2 * (int64_t) f4; + int64_t f1f5_4 = f1_2 * (int64_t) f5_2; + int64_t f1f6_2 = f1_2 * (int64_t) f6; + int64_t f1f7_4 = f1_2 * (int64_t) f7_2; + int64_t f1f8_2 = f1_2 * (int64_t) f8; + int64_t f1f9_76 = f1_2 * (int64_t) f9_38; + int64_t f2f2 = f2 * (int64_t) f2; + int64_t f2f3_2 = f2_2 * (int64_t) f3; + int64_t f2f4_2 = f2_2 * (int64_t) f4; + int64_t f2f5_2 = f2_2 * (int64_t) f5; + int64_t f2f6_2 = f2_2 * (int64_t) f6; + int64_t f2f7_2 = f2_2 * (int64_t) f7; + int64_t f2f8_38 = f2_2 * (int64_t) f8_19; + int64_t f2f9_38 = f2 * (int64_t) f9_38; + int64_t f3f3_2 = f3_2 * (int64_t) f3; + int64_t f3f4_2 = f3_2 * (int64_t) f4; + int64_t f3f5_4 = f3_2 * (int64_t) f5_2; + int64_t f3f6_2 = f3_2 * (int64_t) f6; + int64_t f3f7_76 = f3_2 * (int64_t) f7_38; + int64_t f3f8_38 = f3_2 * (int64_t) f8_19; + int64_t f3f9_76 = f3_2 * (int64_t) f9_38; + int64_t f4f4 = f4 * (int64_t) f4; + int64_t f4f5_2 = f4_2 * (int64_t) f5; + int64_t f4f6_38 = f4_2 * (int64_t) f6_19; + int64_t f4f7_38 = f4 * (int64_t) f7_38; + int64_t f4f8_38 = f4_2 * (int64_t) f8_19; + int64_t f4f9_38 = f4 * (int64_t) f9_38; + int64_t f5f5_38 = f5 * (int64_t) f5_38; + int64_t f5f6_38 = f5_2 * (int64_t) f6_19; + int64_t f5f7_76 = f5_2 * (int64_t) f7_38; + int64_t f5f8_38 = f5_2 * (int64_t) f8_19; + int64_t f5f9_76 = f5_2 * (int64_t) f9_38; + int64_t f6f6_19 = f6 * (int64_t) f6_19; + int64_t f6f7_38 = f6 * (int64_t) f7_38; + int64_t f6f8_38 = f6_2 * (int64_t) f8_19; + int64_t f6f9_38 = f6 * (int64_t) f9_38; + int64_t f7f7_38 = f7 * (int64_t) f7_38; + int64_t f7f8_38 = f7_2 * (int64_t) f8_19; + int64_t f7f9_76 = f7_2 * (int64_t) f9_38; + int64_t f8f8_19 = f8 * (int64_t) f8_19; + int64_t f8f9_38 = f8 * (int64_t) f9_38; + int64_t f9f9_38 = f9 * (int64_t) f9_38; + + int64_t h0 = f0f0 + f1f9_76 + f2f8_38 + f3f7_76 + f4f6_38 + f5f5_38; + int64_t h1 = f0f1_2 + f2f9_38 + f3f8_38 + f4f7_38 + f5f6_38; + int64_t h2 = f0f2_2 + f1f1_2 + f3f9_76 + f4f8_38 + f5f7_76 + f6f6_19; + int64_t h3 = f0f3_2 + f1f2_2 + f4f9_38 + f5f8_38 + f6f7_38; + int64_t h4 = f0f4_2 + f1f3_4 + f2f2 + f5f9_76 + f6f8_38 + f7f7_38; + int64_t h5 = f0f5_2 + f1f4_2 + f2f3_2 + f6f9_38 + f7f8_38; + int64_t h6 = f0f6_2 + f1f5_4 + f2f4_2 + f3f3_2 + f7f9_76 + f8f8_19; + int64_t h7 = f0f7_2 + f1f6_2 + f2f5_2 + f3f4_2 + f8f9_38; + int64_t h8 = f0f8_2 + f1f7_4 + f2f6_2 + f3f5_4 + f4f4 + f9f9_38; + int64_t h9 = f0f9_2 + f1f8_2 + f2f7_2 + f3f6_2 + f4f5_2; + + int64_t carry0; + int64_t carry1; + int64_t carry2; + int64_t carry3; + int64_t carry4; + int64_t carry5; + int64_t carry6; + int64_t carry7; + int64_t carry8; + int64_t carry9; + + h0 += h0; + h1 += h1; + h2 += h2; + h3 += h3; + h4 += h4; + h5 += h5; + h6 += h6; + h7 += h7; + h8 += h8; + h9 += h9; + + carry0 = (h0 + (int64_t)(1L << 25)) >> 26; + h1 += carry0; + h0 -= carry0 * ((uint64_t) 1L << 26); + carry4 = (h4 + (int64_t)(1L << 25)) >> 26; + h5 += carry4; + h4 -= carry4 * ((uint64_t) 1L << 26); + + carry1 = (h1 + (int64_t)(1L << 24)) >> 25; + h2 += carry1; + h1 -= carry1 * ((uint64_t) 1L << 25); + carry5 = (h5 + (int64_t)(1L << 24)) >> 25; + h6 += carry5; + h5 -= carry5 * ((uint64_t) 1L << 25); + + carry2 = (h2 + (int64_t)(1L << 25)) >> 26; + h3 += carry2; + h2 -= carry2 * ((uint64_t) 1L << 26); + carry6 = (h6 + (int64_t)(1L << 25)) >> 26; + h7 += carry6; + h6 -= carry6 * ((uint64_t) 1L << 26); + + carry3 = (h3 + (int64_t)(1L << 24)) >> 25; + h4 += carry3; + h3 -= carry3 * ((uint64_t) 1L << 25); + carry7 = (h7 + (int64_t)(1L << 24)) >> 25; + h8 += carry7; + h7 -= carry7 * ((uint64_t) 1L << 25); + + carry4 = (h4 + (int64_t)(1L << 25)) >> 26; + h5 += carry4; + h4 -= carry4 * ((uint64_t) 1L << 26); + carry8 = (h8 + (int64_t)(1L << 25)) >> 26; + h9 += carry8; + h8 -= carry8 * ((uint64_t) 1L << 26); + + carry9 = (h9 + (int64_t)(1L << 24)) >> 25; + h0 += carry9 * 19; + h9 -= carry9 * ((uint64_t) 1L << 25); + + carry0 = (h0 + (int64_t)(1L << 25)) >> 26; + h1 += carry0; + h0 -= carry0 * ((uint64_t) 1L << 26); + + h[0] = (int32_t) h0; + h[1] = (int32_t) h1; + h[2] = (int32_t) h2; + h[3] = (int32_t) h3; + h[4] = (int32_t) h4; + h[5] = (int32_t) h5; + h[6] = (int32_t) h6; + h[7] = (int32_t) h7; + h[8] = (int32_t) h8; + h[9] = (int32_t) h9; +} + +static void +fe25519_scalar_product(fe25519 h, const fe25519 f, uint32_t n) +{ + int64_t sn = (int64_t) n; + int32_t f0 = f[0]; + int32_t f1 = f[1]; + int32_t f2 = f[2]; + int32_t f3 = f[3]; + int32_t f4 = f[4]; + int32_t f5 = f[5]; + int32_t f6 = f[6]; + int32_t f7 = f[7]; + int32_t f8 = f[8]; + int32_t f9 = f[9]; + int64_t h0 = f0 * sn; + int64_t h1 = f1 * sn; + int64_t h2 = f2 * sn; + int64_t h3 = f3 * sn; + int64_t h4 = f4 * sn; + int64_t h5 = f5 * sn; + int64_t h6 = f6 * sn; + int64_t h7 = f7 * sn; + int64_t h8 = f8 * sn; + int64_t h9 = f9 * sn; + int64_t carry0, carry1, carry2, carry3, carry4, carry5, carry6, carry7, + carry8, carry9; + + carry9 = (h9 + ((int64_t) 1 << 24)) >> 25; + h0 += carry9 * 19; + h9 -= carry9 * ((int64_t) 1 << 25); + carry1 = (h1 + ((int64_t) 1 << 24)) >> 25; + h2 += carry1; + h1 -= carry1 * ((int64_t) 1 << 25); + carry3 = (h3 + ((int64_t) 1 << 24)) >> 25; + h4 += carry3; + h3 -= carry3 * ((int64_t) 1 << 25); + carry5 = (h5 + ((int64_t) 1 << 24)) >> 25; + h6 += carry5; + h5 -= carry5 * ((int64_t) 1 << 25); + carry7 = (h7 + ((int64_t) 1 << 24)) >> 25; + h8 += carry7; + h7 -= carry7 * ((int64_t) 1 << 25); + + carry0 = (h0 + ((int64_t) 1 << 25)) >> 26; + h1 += carry0; + h0 -= carry0 * ((int64_t) 1 << 26); + carry2 = (h2 + ((int64_t) 1 << 25)) >> 26; + h3 += carry2; + h2 -= carry2 * ((int64_t) 1 << 26); + carry4 = (h4 + ((int64_t) 1 << 25)) >> 26; + h5 += carry4; + h4 -= carry4 * ((int64_t) 1 << 26); + carry6 = (h6 + ((int64_t) 1 << 25)) >> 26; + h7 += carry6; + h6 -= carry6 * ((int64_t) 1 << 26); + carry8 = (h8 + ((int64_t) 1 << 25)) >> 26; + h9 += carry8; + h8 -= carry8 * ((int64_t) 1 << 26); + + h[0] = (int32_t) h0; + h[1] = (int32_t) h1; + h[2] = (int32_t) h2; + h[3] = (int32_t) h3; + h[4] = (int32_t) h4; + h[5] = (int32_t) h5; + h[6] = (int32_t) h6; + h[7] = (int32_t) h7; + h[8] = (int32_t) h8; + h[9] = (int32_t) h9; +} + +#if defined(__ORBIS__) || defined(__PROSPERO__) +#pragma clang diagnostic pop +#endif // #if defined(__ORBIS__) || defined(__PROSPERO__) + diff --git a/sodium/sodium_private_ed25519_ref10_fe_51.h b/sodium/sodium_private_ed25519_ref10_fe_51.h new file mode 100644 index 00000000..cf85d368 --- /dev/null +++ b/sodium/sodium_private_ed25519_ref10_fe_51.h @@ -0,0 +1,518 @@ +#include + +#include "sodium_private_common.h" +#include "sodium_utils.h" + +/* + h = 0 + */ + +static inline void +fe25519_0(fe25519 h) +{ + memset(&h[0], 0, 5 * sizeof h[0]); +} + +/* + h = 1 + */ + +static inline void +fe25519_1(fe25519 h) +{ + h[0] = 1; + memset(&h[1], 0, 4 * sizeof h[0]); +} + +/* + h = f + g + Can overlap h with f or g. + */ + +static inline void +fe25519_add(fe25519 h, const fe25519 f, const fe25519 g) +{ + uint64_t h0 = f[0] + g[0]; + uint64_t h1 = f[1] + g[1]; + uint64_t h2 = f[2] + g[2]; + uint64_t h3 = f[3] + g[3]; + uint64_t h4 = f[4] + g[4]; + + h[0] = h0; + h[1] = h1; + h[2] = h2; + h[3] = h3; + h[4] = h4; +} + +/* + h = f - g + */ + +static void +fe25519_sub(fe25519 h, const fe25519 f, const fe25519 g) +{ + const uint64_t mask = 0x7ffffffffffffULL; + uint64_t h0, h1, h2, h3, h4; + + h0 = g[0]; + h1 = g[1]; + h2 = g[2]; + h3 = g[3]; + h4 = g[4]; + + h1 += h0 >> 51; + h0 &= mask; + h2 += h1 >> 51; + h1 &= mask; + h3 += h2 >> 51; + h2 &= mask; + h4 += h3 >> 51; + h3 &= mask; + h0 += 19ULL * (h4 >> 51); + h4 &= mask; + + h0 = (f[0] + 0xfffffffffffdaULL) - h0; + h1 = (f[1] + 0xffffffffffffeULL) - h1; + h2 = (f[2] + 0xffffffffffffeULL) - h2; + h3 = (f[3] + 0xffffffffffffeULL) - h3; + h4 = (f[4] + 0xffffffffffffeULL) - h4; + + h[0] = h0; + h[1] = h1; + h[2] = h2; + h[3] = h3; + h[4] = h4; +} + +/* + h = -f + */ + +static inline void +fe25519_neg(fe25519 h, const fe25519 f) +{ + fe25519 zero; + + fe25519_0(zero); + fe25519_sub(h, zero, f); +} + +/* + Replace (f,g) with (g,g) if b == 1; + replace (f,g) with (f,g) if b == 0. + * + Preconditions: b in {0,1}. + */ + +static void +fe25519_cmov(fe25519 f, const fe25519 g, unsigned int b) +{ + const uint64_t mask = (uint64_t) (-(int64_t) b); + + uint64_t f0 = f[0]; + uint64_t f1 = f[1]; + uint64_t f2 = f[2]; + uint64_t f3 = f[3]; + uint64_t f4 = f[4]; + + uint64_t x0 = f0 ^ g[0]; + uint64_t x1 = f1 ^ g[1]; + uint64_t x2 = f2 ^ g[2]; + uint64_t x3 = f3 ^ g[3]; + uint64_t x4 = f4 ^ g[4]; + + x0 &= mask; + x1 &= mask; + x2 &= mask; + x3 &= mask; + x4 &= mask; + + f[0] = f0 ^ x0; + f[1] = f1 ^ x1; + f[2] = f2 ^ x2; + f[3] = f3 ^ x3; + f[4] = f4 ^ x4; +} + +/* +Replace (f,g) with (g,f) if b == 1; +replace (f,g) with (f,g) if b == 0. + +Preconditions: b in {0,1}. +*/ + +static void +fe25519_cswap(fe25519 f, fe25519 g, unsigned int b) +{ + const uint64_t mask = (uint64_t) (-(int64_t) b); + + uint64_t f0 = f[0]; + uint64_t f1 = f[1]; + uint64_t f2 = f[2]; + uint64_t f3 = f[3]; + uint64_t f4 = f[4]; + + uint64_t g0 = g[0]; + uint64_t g1 = g[1]; + uint64_t g2 = g[2]; + uint64_t g3 = g[3]; + uint64_t g4 = g[4]; + + uint64_t x0 = f0 ^ g0; + uint64_t x1 = f1 ^ g1; + uint64_t x2 = f2 ^ g2; + uint64_t x3 = f3 ^ g3; + uint64_t x4 = f4 ^ g4; + + x0 &= mask; + x1 &= mask; + x2 &= mask; + x3 &= mask; + x4 &= mask; + + f[0] = f0 ^ x0; + f[1] = f1 ^ x1; + f[2] = f2 ^ x2; + f[3] = f3 ^ x3; + f[4] = f4 ^ x4; + + g[0] = g0 ^ x0; + g[1] = g1 ^ x1; + g[2] = g2 ^ x2; + g[3] = g3 ^ x3; + g[4] = g4 ^ x4; +} + +/* + h = f + */ + +static inline void +fe25519_copy(fe25519 h, const fe25519 f) +{ + uint64_t f0 = f[0]; + uint64_t f1 = f[1]; + uint64_t f2 = f[2]; + uint64_t f3 = f[3]; + uint64_t f4 = f[4]; + + h[0] = f0; + h[1] = f1; + h[2] = f2; + h[3] = f3; + h[4] = f4; +} + +/* + return 1 if f is in {1,3,5,...,q-2} + return 0 if f is in {0,2,4,...,q-1} + */ + +static inline int +fe25519_isnegative(const fe25519 f) +{ + unsigned char s[32]; + + fe25519_tobytes(s, f); + + return s[0] & 1; +} + +/* + return 1 if f == 0 + return 0 if f != 0 + */ + +static inline int +fe25519_iszero(const fe25519 f) +{ + unsigned char s[32]; + + fe25519_tobytes(s, f); + + return sodium_is_zero(s, 32); +} + +/* + h = f * g + Can overlap h with f or g. + */ + +static void +fe25519_mul(fe25519 h, const fe25519 f, const fe25519 g) +{ + const uint64_t mask = 0x7ffffffffffffULL; + uint128_t r0, r1, r2, r3, r4, carry; + uint64_t f0, f1, f2, f3, f4; + uint64_t f1_19, f2_19, f3_19, f4_19; + uint64_t g0, g1, g2, g3, g4; + uint64_t r00, r01, r02, r03, r04; + + f0 = f[0]; + f1 = f[1]; + f2 = f[2]; + f3 = f[3]; + f4 = f[4]; + + g0 = g[0]; + g1 = g[1]; + g2 = g[2]; + g3 = g[3]; + g4 = g[4]; + + f1_19 = 19ULL * f1; + f2_19 = 19ULL * f2; + f3_19 = 19ULL * f3; + f4_19 = 19ULL * f4; + + r0 = ((uint128_t) f0 ) * ((uint128_t) g0); + r0 += ((uint128_t) f1_19) * ((uint128_t) g4); + r0 += ((uint128_t) f2_19) * ((uint128_t) g3); + r0 += ((uint128_t) f3_19) * ((uint128_t) g2); + r0 += ((uint128_t) f4_19) * ((uint128_t) g1); + + r1 = ((uint128_t) f0 ) * ((uint128_t) g1); + r1 += ((uint128_t) f1 ) * ((uint128_t) g0); + r1 += ((uint128_t) f2_19) * ((uint128_t) g4); + r1 += ((uint128_t) f3_19) * ((uint128_t) g3); + r1 += ((uint128_t) f4_19) * ((uint128_t) g2); + + r2 = ((uint128_t) f0 ) * ((uint128_t) g2); + r2 += ((uint128_t) f1 ) * ((uint128_t) g1); + r2 += ((uint128_t) f2 ) * ((uint128_t) g0); + r2 += ((uint128_t) f3_19) * ((uint128_t) g4); + r2 += ((uint128_t) f4_19) * ((uint128_t) g3); + + r3 = ((uint128_t) f0 ) * ((uint128_t) g3); + r3 += ((uint128_t) f1 ) * ((uint128_t) g2); + r3 += ((uint128_t) f2 ) * ((uint128_t) g1); + r3 += ((uint128_t) f3 ) * ((uint128_t) g0); + r3 += ((uint128_t) f4_19) * ((uint128_t) g4); + + r4 = ((uint128_t) f0 ) * ((uint128_t) g4); + r4 += ((uint128_t) f1 ) * ((uint128_t) g3); + r4 += ((uint128_t) f2 ) * ((uint128_t) g2); + r4 += ((uint128_t) f3 ) * ((uint128_t) g1); + r4 += ((uint128_t) f4 ) * ((uint128_t) g0); + + r00 = ((uint64_t) r0) & mask; + carry = r0 >> 51; + r1 += carry; + r01 = ((uint64_t) r1) & mask; + carry = r1 >> 51; + r2 += carry; + r02 = ((uint64_t) r2) & mask; + carry = r2 >> 51; + r3 += carry; + r03 = ((uint64_t) r3) & mask; + carry = r3 >> 51; + r4 += carry; + r04 = ((uint64_t) r4) & mask; + carry = r4 >> 51; + r00 += 19ULL * (uint64_t) carry; + carry = r00 >> 51; + r00 &= mask; + r01 += (uint64_t) carry; + carry = r01 >> 51; + r01 &= mask; + r02 += (uint64_t) carry; + + h[0] = r00; + h[1] = r01; + h[2] = r02; + h[3] = r03; + h[4] = r04; +} + +/* + h = f * f + Can overlap h with f. + */ + +static void +fe25519_sq(fe25519 h, const fe25519 f) +{ + const uint64_t mask = 0x7ffffffffffffULL; + uint128_t r0, r1, r2, r3, r4, carry; + uint64_t f0, f1, f2, f3, f4; + uint64_t f0_2, f1_2, f1_38, f2_38, f3_38, f3_19, f4_19; + uint64_t r00, r01, r02, r03, r04; + + f0 = f[0]; + f1 = f[1]; + f2 = f[2]; + f3 = f[3]; + f4 = f[4]; + + f0_2 = f0 << 1; + f1_2 = f1 << 1; + + f1_38 = 38ULL * f1; + f2_38 = 38ULL * f2; + f3_38 = 38ULL * f3; + + f3_19 = 19ULL * f3; + f4_19 = 19ULL * f4; + + r0 = ((uint128_t) f0 ) * ((uint128_t) f0); + r0 += ((uint128_t) f1_38) * ((uint128_t) f4); + r0 += ((uint128_t) f2_38) * ((uint128_t) f3); + + r1 = ((uint128_t) f0_2 ) * ((uint128_t) f1); + r1 += ((uint128_t) f2_38) * ((uint128_t) f4); + r1 += ((uint128_t) f3_19) * ((uint128_t) f3); + + r2 = ((uint128_t) f0_2 ) * ((uint128_t) f2); + r2 += ((uint128_t) f1 ) * ((uint128_t) f1); + r2 += ((uint128_t) f3_38) * ((uint128_t) f4); + + r3 = ((uint128_t) f0_2 ) * ((uint128_t) f3); + r3 += ((uint128_t) f1_2 ) * ((uint128_t) f2); + r3 += ((uint128_t) f4_19) * ((uint128_t) f4); + + r4 = ((uint128_t) f0_2 ) * ((uint128_t) f4); + r4 += ((uint128_t) f1_2 ) * ((uint128_t) f3); + r4 += ((uint128_t) f2 ) * ((uint128_t) f2); + + r00 = ((uint64_t) r0) & mask; + carry = r0 >> 51; + r1 += carry; + r01 = ((uint64_t) r1) & mask; + carry = r1 >> 51; + r2 += carry; + r02 = ((uint64_t) r2) & mask; + carry = r2 >> 51; + r3 += carry; + r03 = ((uint64_t) r3) & mask; + carry = r3 >> 51; + r4 += carry; + r04 = ((uint64_t) r4) & mask; + carry = r4 >> 51; + r00 += 19ULL * (uint64_t) carry; + carry = r00 >> 51; + r00 &= mask; + r01 += (uint64_t) carry; + carry = r01 >> 51; + r01 &= mask; + r02 += (uint64_t) carry; + + h[0] = r00; + h[1] = r01; + h[2] = r02; + h[3] = r03; + h[4] = r04; +} + +/* + h = 2 * f * f + Can overlap h with f. +*/ + +static void +fe25519_sq2(fe25519 h, const fe25519 f) +{ + const uint64_t mask = 0x7ffffffffffffULL; + uint128_t r0, r1, r2, r3, r4, carry; + uint64_t f0, f1, f2, f3, f4; + uint64_t f0_2, f1_2, f1_38, f2_38, f3_38, f3_19, f4_19; + uint64_t r00, r01, r02, r03, r04; + + f0 = f[0]; + f1 = f[1]; + f2 = f[2]; + f3 = f[3]; + f4 = f[4]; + + f0_2 = f0 << 1; + f1_2 = f1 << 1; + + f1_38 = 38ULL * f1; + f2_38 = 38ULL * f2; + f3_38 = 38ULL * f3; + + f3_19 = 19ULL * f3; + f4_19 = 19ULL * f4; + + r0 = ((uint128_t) f0 ) * ((uint128_t) f0); + r0 += ((uint128_t) f1_38) * ((uint128_t) f4); + r0 += ((uint128_t) f2_38) * ((uint128_t) f3); + + r1 = ((uint128_t) f0_2 ) * ((uint128_t) f1); + r1 += ((uint128_t) f2_38) * ((uint128_t) f4); + r1 += ((uint128_t) f3_19) * ((uint128_t) f3); + + r2 = ((uint128_t) f0_2 ) * ((uint128_t) f2); + r2 += ((uint128_t) f1 ) * ((uint128_t) f1); + r2 += ((uint128_t) f3_38) * ((uint128_t) f4); + + r3 = ((uint128_t) f0_2 ) * ((uint128_t) f3); + r3 += ((uint128_t) f1_2 ) * ((uint128_t) f2); + r3 += ((uint128_t) f4_19) * ((uint128_t) f4); + + r4 = ((uint128_t) f0_2 ) * ((uint128_t) f4); + r4 += ((uint128_t) f1_2 ) * ((uint128_t) f3); + r4 += ((uint128_t) f2 ) * ((uint128_t) f2); + + r0 <<= 1; + r1 <<= 1; + r2 <<= 1; + r3 <<= 1; + r4 <<= 1; + + r00 = ((uint64_t) r0) & mask; + carry = r0 >> 51; + r1 += carry; + r01 = ((uint64_t) r1) & mask; + carry = r1 >> 51; + r2 += carry; + r02 = ((uint64_t) r2) & mask; + carry = r2 >> 51; + r3 += carry; + r03 = ((uint64_t) r3) & mask; + carry = r3 >> 51; + r4 += carry; + r04 = ((uint64_t) r4) & mask; + carry = r4 >> 51; + r00 += 19ULL * (uint64_t) carry; + carry = r00 >> 51; + r00 &= mask; + r01 += (uint64_t) carry; + carry = r01 >> 51; + r01 &= mask; + r02 += (uint64_t) carry; + + h[0] = r00; + h[1] = r01; + h[2] = r02; + h[3] = r03; + h[4] = r04; +} + +static void +fe25519_scalar_product(fe25519 h, const fe25519 f, uint32_t n) +{ + const uint64_t mask = 0x7ffffffffffffULL; + uint128_t a; + uint128_t sn = (uint128_t) n; + uint64_t h0, h1, h2, h3, h4; + + a = f[0] * sn; + h0 = ((uint64_t) a) & mask; + a = f[1] * sn + ((uint64_t) (a >> 51)); + h1 = ((uint64_t) a) & mask; + a = f[2] * sn + ((uint64_t) (a >> 51)); + h2 = ((uint64_t) a) & mask; + a = f[3] * sn + ((uint64_t) (a >> 51)); + h3 = ((uint64_t) a) & mask; + a = f[4] * sn + ((uint64_t) (a >> 51)); + h4 = ((uint64_t) a) & mask; + + h0 += (a >> 51) * 19ULL; + + h[0] = h0; + h[1] = h1; + h[2] = h2; + h[3] = h3; + h[4] = h4; +} diff --git a/sodium/sodium_private_implementations.h b/sodium/sodium_private_implementations.h new file mode 100644 index 00000000..91c8be57 --- /dev/null +++ b/sodium/sodium_private_implementations.h @@ -0,0 +1,10 @@ +#ifndef implementations_H +#define implementations_H + +int _crypto_generichash_blake2b_pick_best_implementation(void); +int _crypto_onetimeauth_poly1305_pick_best_implementation(void); +int _crypto_scalarmult_curve25519_pick_best_implementation(void); +int _crypto_stream_chacha20_pick_best_implementation(void); +int _crypto_stream_salsa20_pick_best_implementation(void); + +#endif diff --git a/sodium/sodium_private_mutex.h b/sodium/sodium_private_mutex.h new file mode 100644 index 00000000..322b6742 --- /dev/null +++ b/sodium/sodium_private_mutex.h @@ -0,0 +1,7 @@ +#ifndef mutex_H +#define mutex_H 1 + +extern int sodium_crit_enter(void); +extern int sodium_crit_leave(void); + +#endif diff --git a/sodium/sodium_private_sse2_64_32.h b/sodium/sodium_private_sse2_64_32.h new file mode 100644 index 00000000..73a63d2d --- /dev/null +++ b/sodium/sodium_private_sse2_64_32.h @@ -0,0 +1,50 @@ +#ifndef sse2_64_32_H +#define sse2_64_32_H 1 + +#include "sodium_private_common.h" + +#ifdef HAVE_INTRIN_H +# include +#endif + +#if defined(HAVE_EMMINTRIN_H) && \ + !(defined(__amd64) || defined(__amd64__) || defined(__x86_64__) || \ + defined(_M_X64) || defined(_M_AMD64)) + +# include +# include + +# ifndef _mm_set_epi64x +# define _mm_set_epi64x(Q0, Q1) sodium__mm_set_epi64x((Q0), (Q1)) +static inline __m128i +sodium__mm_set_epi64x(int64_t q1, int64_t q0) +{ + union { int64_t as64; int32_t as32[2]; } x0, x1; + x0.as64 = q0; x1.as64 = q1; + return _mm_set_epi32(x1.as32[1], x1.as32[0], x0.as32[1], x0.as32[0]); +} +# endif + +# ifndef _mm_set1_epi64x +# define _mm_set1_epi64x(Q) sodium__mm_set1_epi64x(Q) +static inline __m128i +sodium__mm_set1_epi64x(int64_t q) +{ + return _mm_set_epi64x(q, q); +} +# endif + +# ifndef _mm_cvtsi64_si128 +# define _mm_cvtsi64_si128(Q) sodium__mm_cvtsi64_si128(Q) +static inline __m128i +sodium__mm_cvtsi64_si128(int64_t q) +{ + union { int64_t as64; int32_t as32[2]; } x; + x.as64 = q; + return _mm_setr_epi32(x.as32[0], x.as32[1], 0, 0); +} +# endif + +#endif + +#endif diff --git a/sodium/sodium_randombytes.c b/sodium/sodium_randombytes.c new file mode 100644 index 00000000..f6d3b222 --- /dev/null +++ b/sodium/sodium_randombytes.c @@ -0,0 +1,207 @@ + +#include +#include +#include +#include + +#include + +#ifdef __EMSCRIPTEN__ +# include +#endif + +#include "sodium_core.h" +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_randombytes.h" +#ifdef RANDOMBYTES_DEFAULT_IMPLEMENTATION +# include "sodium_randombytes_default.h" +#else +# ifdef __native_client__ +# include "sodium_randombytes_nativeclient.h" +# else +# include "sodium_randombytes_sysrandom.h" +# endif +#endif +#include "sodium_private_common.h" +#include + +/* C++Builder defines a "random" macro */ +#undef random + +static const randombytes_implementation *implementation; + +#ifndef RANDOMBYTES_DEFAULT_IMPLEMENTATION +# ifdef __EMSCRIPTEN__ +# define RANDOMBYTES_DEFAULT_IMPLEMENTATION NULL +# else +# ifdef __native_client__ +# define RANDOMBYTES_DEFAULT_IMPLEMENTATION &randombytes_nativeclient_implementation; +# else +# define RANDOMBYTES_DEFAULT_IMPLEMENTATION &randombytes_sysrandom_implementation; +# endif +# endif +#endif + +static void +randombytes_init_if_needed(void) +{ + if (implementation == NULL) { + implementation = RANDOMBYTES_DEFAULT_IMPLEMENTATION; + randombytes_stir(); + } +} + +int +randombytes_set_implementation(randombytes_implementation *impl) +{ + implementation = impl; + + return 0; +} + +const char * +randombytes_implementation_name(void) +{ +#ifndef __EMSCRIPTEN__ + randombytes_init_if_needed(); + return implementation->implementation_name(); +#else + return "js"; +#endif +} + +uint32_t +randombytes_random(void) +{ +#ifndef __EMSCRIPTEN__ + randombytes_init_if_needed(); + return implementation->random(); +#else + return EM_ASM_INT_V({ + return Module.getRandomValue(); + }); +#endif +} + +void +randombytes_stir(void) +{ +#ifndef __EMSCRIPTEN__ + randombytes_init_if_needed(); + if (implementation->stir != NULL) { + implementation->stir(); + } +#else + EM_ASM({ + if (Module.getRandomValue === undefined) { + try { + var window_ = 'object' === typeof window ? window : self; + var crypto_ = typeof window_.crypto !== 'undefined' ? window_.crypto : window_.msCrypto; + var randomValuesStandard = function() { + var buf = new Uint32Array(1); + crypto_.getRandomValues(buf); + return buf[0] >>> 0; + }; + randomValuesStandard(); + Module.getRandomValue = randomValuesStandard; + } catch (e) { + try { + var crypto = require('crypto'); + var randomValueNodeJS = function() { + var buf = crypto['randomBytes'](4); + return (buf[0] << 24 | buf[1] << 16 | buf[2] << 8 | buf[3]) >>> 0; + }; + randomValueNodeJS(); + Module.getRandomValue = randomValueNodeJS; + } catch (e) { + throw 'No secure random number generator found'; + } + } + } + }); +#endif +} + +uint32_t +randombytes_uniform(const uint32_t upper_bound) +{ + uint32_t min; + uint32_t r; + +#ifndef __EMSCRIPTEN__ + randombytes_init_if_needed(); + if (implementation->uniform != NULL) { + return implementation->uniform(upper_bound); + } +#endif + if (upper_bound < 2) { + return 0; + } + min = (1U + ~upper_bound) % upper_bound; /* = 2**32 mod upper_bound */ + do { + r = randombytes_random(); + } while (r < min); + /* r is now clamped to a set whose size mod upper_bound == 0 + * the worst case (2**31+1) requires ~ 2 attempts */ + + return r % upper_bound; +} + +void +randombytes_buf(void * const buf, const size_t size) +{ +#ifndef __EMSCRIPTEN__ + randombytes_init_if_needed(); + if (size > (size_t) 0U) { + implementation->buf(buf, size); + } +#else + unsigned char *p = (unsigned char *) buf; + size_t i; + + for (i = (size_t) 0U; i < size; i++) { + p[i] = (unsigned char) randombytes_random(); + } +#endif +} + +void +randombytes_buf_deterministic(void * const buf, const size_t size, + const unsigned char seed[randombytes_SEEDBYTES]) +{ + static const unsigned char nonce[crypto_stream_chacha20_ietf_NONCEBYTES] = { + 'L', 'i', 'b', 's', 'o', 'd', 'i', 'u', 'm', 'D', 'R', 'G' + }; + + COMPILER_ASSERT(randombytes_SEEDBYTES == crypto_stream_chacha20_ietf_KEYBYTES); +#if SIZE_MAX > 0x4000000000ULL + COMPILER_ASSERT(randombytes_BYTES_MAX <= 0x4000000000ULL); + if (size > 0x4000000000ULL) { + sodium_misuse(); + } +#endif + crypto_stream_chacha20_ietf((unsigned char *) buf, (unsigned long long) size, + nonce, seed); +} + +size_t +randombytes_seedbytes(void) +{ + return randombytes_SEEDBYTES; +} + +int +randombytes_close(void) +{ + if (implementation != NULL && implementation->close != NULL) { + return implementation->close(); + } + return 0; +} + +void +randombytes(unsigned char * const buf, const unsigned long long buf_len) +{ + assert(buf_len <= SIZE_MAX); + randombytes_buf(buf, (size_t) buf_len); +} diff --git a/windows/sodium/randombytes.h b/sodium/sodium_randombytes.h similarity index 79% rename from windows/sodium/randombytes.h rename to sodium/sodium_randombytes.h index d112fb29..45ef9ae6 100644 --- a/windows/sodium/randombytes.h +++ b/sodium/sodium_randombytes.h @@ -7,7 +7,7 @@ #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus # ifdef __GNUC__ @@ -25,39 +25,35 @@ typedef struct randombytes_implementation { int (*close)(void); /* optional */ } randombytes_implementation; +#define randombytes_BYTES_MAX SODIUM_MIN(SODIUM_SIZE_MAX, 0xffffffffUL) + #define randombytes_SEEDBYTES 32U -SODIUM_EXPORT size_t randombytes_seedbytes(void); -SODIUM_EXPORT -void randombytes_buf(void * const buf, const size_t size); +void randombytes_buf(void * const buf, const size_t size) + __attribute__ ((nonnull)); -SODIUM_EXPORT void randombytes_buf_deterministic(void * const buf, const size_t size, - const unsigned char seed[randombytes_SEEDBYTES]); + const unsigned char seed[randombytes_SEEDBYTES]) + __attribute__ ((nonnull)); -SODIUM_EXPORT uint32_t randombytes_random(void); -SODIUM_EXPORT uint32_t randombytes_uniform(const uint32_t upper_bound); -SODIUM_EXPORT void randombytes_stir(void); -SODIUM_EXPORT int randombytes_close(void); -SODIUM_EXPORT -int randombytes_set_implementation(randombytes_implementation *impl); +int randombytes_set_implementation(randombytes_implementation *impl) + __attribute__ ((nonnull)); -SODIUM_EXPORT const char *randombytes_implementation_name(void); /* -- NaCl compatibility interface -- */ -SODIUM_EXPORT -void randombytes(unsigned char * const buf, const unsigned long long buf_len); +void randombytes(unsigned char * const buf, const unsigned long long buf_len) + __attribute__ ((nonnull)); #ifdef __cplusplus } diff --git a/sodium/sodium_randombytes_nativeclient.h b/sodium/sodium_randombytes_nativeclient.h new file mode 100644 index 00000000..70457ff5 --- /dev/null +++ b/sodium/sodium_randombytes_nativeclient.h @@ -0,0 +1,22 @@ + +#ifndef randombytes_nativeclient_H +#define randombytes_nativeclient_H + +#ifdef __native_client__ + +# include "sodium_export.h" +# include "sodium_randombytes.h" + +# ifdef __cplusplus +extern "C" { +# endif + +extern struct randombytes_implementation randombytes_nativeclient_implementation; + +# ifdef __cplusplus +} +# endif + +#endif + +#endif diff --git a/sodium/sodium_randombytes_salsa20_random.c b/sodium/sodium_randombytes_salsa20_random.c new file mode 100644 index 00000000..2cf65f5d --- /dev/null +++ b/sodium/sodium_randombytes_salsa20_random.c @@ -0,0 +1,568 @@ + +#include +#include +#include +#include +#include +#include +#include +#include +#if !defined(_MSC_VER) && !defined(__BORLANDC__) +# include +#endif + +#include +#ifndef _WIN32 +# include +# include +#endif +#ifdef __linux__ +# ifdef __dietlibc__ +# define _LINUX_SOURCE +# include +# define HAVE_LINUX_COMPATIBLE_GETRANDOM +# else /* __dietlibc__ */ +# include +# if defined(SYS_getrandom) && defined(__NR_getrandom) +# define getrandom(B, S, F) syscall(SYS_getrandom, (B), (int) (S), (F)) +# define HAVE_LINUX_COMPATIBLE_GETRANDOM +# endif +# endif /* __dietlibc__ */ +#elif defined(__FreeBSD__) +# include +# if defined(__FreeBSD_version) && __FreeBSD_version >= 1200000 +# include +# define HAVE_LINUX_COMPATIBLE_GETRANDOM +# endif +#endif +#if !defined(NO_BLOCKING_RANDOM_POLL) && defined(__linux__) +# define BLOCK_ON_DEV_RANDOM +#endif +#ifdef BLOCK_ON_DEV_RANDOM +# include +#endif +#ifdef HAVE_RDRAND +# pragma GCC target("rdrnd") +# include +#endif + +#include "sodium_core.h" +#include "sodium_crypto_core_salsa20.h" +#include "sodium_crypto_stream_salsa20.h" +#include "sodium_private_common.h" +#include "sodium_randombytes.h" +#include "sodium_randombytes_salsa20_random.h" +#include "sodium_runtime.h" +#include "sodium_utils.h" + +#ifdef _WIN32 +# include +# include +# define RtlGenRandom SystemFunction036 +# if defined(__cplusplus) +extern "C" +# endif +#if !defined(_XBOX_ONE) && !defined(_GAMING_XBOX) +BOOLEAN NTAPI RtlGenRandom(PVOID RandomBuffer, ULONG RandomBufferLength); +# pragma comment(lib, "advapi32.lib") +#endif +# ifdef __BORLANDC__ +# define _ftime ftime +# define _timeb timeb +# endif +#endif + +#define SALSA20_RANDOM_BLOCK_SIZE crypto_core_salsa20_OUTPUTBYTES + +#if defined(__OpenBSD__) || defined(__CloudABI__) +# define HAVE_SAFE_ARC4RANDOM 1 +#endif + +#ifndef SSIZE_MAX +# define SSIZE_MAX (SIZE_MAX / 2 - 1) +#endif +#ifndef S_ISNAM +# ifdef __COMPCERT__ +# define S_ISNAM(X) 1 +# else +# define S_ISNAM(X) 0 +# endif +#endif + +#ifndef TLS +# ifdef _WIN32 +# define TLS __declspec(thread) +# else +# define TLS +# endif +#endif + +typedef struct Salsa20RandomGlobal_ { + int initialized; + int random_data_source_fd; + int getrandom_available; + int rdrand_available; +#ifdef HAVE_GETPID + pid_t pid; +#endif +} Salsa20RandomGlobal; + +typedef struct Salsa20Random_ { + int initialized; + size_t rnd32_outleft; + unsigned char key[crypto_stream_salsa20_KEYBYTES]; + unsigned char rnd32[16U * SALSA20_RANDOM_BLOCK_SIZE]; + uint64_t nonce; +} Salsa20Random; + +static Salsa20RandomGlobal global; + +static TLS Salsa20Random stream; + +/* + * Get a high-resolution timestamp, as a uint64_t value + */ + +#ifdef _WIN32 +static uint64_t +sodium_hrtime(void) +{ + struct _timeb tb; +# pragma warning(push) +# pragma warning(disable: 4996) + _ftime(&tb); +# pragma warning(pop) + return ((uint64_t) tb.time) * 1000000U + ((uint64_t) tb.millitm) * 1000U; +} + +#else /* _WIN32 */ + +static uint64_t +sodium_hrtime(void) +{ + struct timeval tv; + + if (gettimeofday(&tv, NULL) != 0) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } + return ((uint64_t) tv.tv_sec) * 1000000U + (uint64_t) tv.tv_usec; +} +#endif + +/* + * Initialize the entropy source + */ + +#ifdef _WIN32 + +static void +randombytes_salsa20_random_init(void) +{ + global.rdrand_available = sodium_runtime_has_rdrand(); +} + +#else /* _WIN32 */ + +static ssize_t +safe_read(const int fd, void * const buf_, size_t size) +{ + unsigned char *buf = (unsigned char *) buf_; + ssize_t readnb; + + assert(size > (size_t) 0U); + assert(size <= SSIZE_MAX); + do { + while ((readnb = read(fd, buf, size)) < (ssize_t) 0 && + (errno == EINTR || errno == EAGAIN)); /* LCOV_EXCL_LINE */ + if (readnb < (ssize_t) 0) { + return readnb; /* LCOV_EXCL_LINE */ + } + if (readnb == (ssize_t) 0) { + break; /* LCOV_EXCL_LINE */ + } + size -= (size_t) readnb; + buf += readnb; + } while (size > (ssize_t) 0); + + return (ssize_t) (buf - (unsigned char *) buf_); +} + +# ifdef BLOCK_ON_DEV_RANDOM +static int +randombytes_block_on_dev_random(void) +{ + struct pollfd pfd; + int fd; + int pret; + + fd = open("/dev/random", O_RDONLY); + if (fd == -1) { + return 0; + } + pfd.fd = fd; + pfd.events = POLLIN; + pfd.revents = 0; + do { + pret = poll(&pfd, 1, -1); + } while (pret < 0 && (errno == EINTR || errno == EAGAIN)); + if (pret != 1) { + (void) close(fd); + errno = EIO; + return -1; + } + return close(fd); +} +# endif + +# ifndef HAVE_SAFE_ARC4RANDOM +static int +randombytes_salsa20_random_random_dev_open(void) +{ +/* LCOV_EXCL_START */ + struct stat st; + static const char *devices[] = { +# ifndef USE_BLOCKING_RANDOM + "/dev/urandom", +# endif + "/dev/random", NULL + }; + const char **device = devices; + int fd; + +# ifdef BLOCK_ON_DEV_RANDOM + if (randombytes_block_on_dev_random() != 0) { + return -1; + } +# endif + do { + fd = open(*device, O_RDONLY); + if (fd != -1) { + if (fstat(fd, &st) == 0 && (S_ISNAM(st.st_mode) || S_ISCHR(st.st_mode))) { +# if defined(F_SETFD) && defined(FD_CLOEXEC) && !defined(NN_NINTENDO_SDK) + (void) fcntl(fd, F_SETFD, fcntl(fd, F_GETFD) | FD_CLOEXEC); +# endif + return fd; + } + (void) close(fd); + } else if (errno == EINTR) { + continue; + } + device++; + } while (*device != NULL); + + errno = EIO; + return -1; +/* LCOV_EXCL_STOP */ +} +# endif + +# ifdef HAVE_LINUX_COMPATIBLE_GETRANDOM +static int +_randombytes_linux_getrandom(void * const buf, const size_t size) +{ + int readnb; + + assert(size <= 256U); + do { + readnb = getrandom(buf, size, 0); + } while (readnb < 0 && (errno == EINTR || errno == EAGAIN)); + + return (readnb == (int) size) - 1; +} + +static int +randombytes_linux_getrandom(void * const buf_, size_t size) +{ + unsigned char *buf = (unsigned char *) buf_; + size_t chunk_size = 256U; + + do { + if (size < chunk_size) { + chunk_size = size; + assert(chunk_size > (size_t) 0U); + } + if (_randombytes_linux_getrandom(buf, chunk_size) != 0) { + return -1; + } + size -= chunk_size; + buf += chunk_size; + } while (size > (size_t) 0U); + + return 0; +} +# endif + +static void +randombytes_salsa20_random_init(void) +{ + const int errno_save = errno; + + global.rdrand_available = sodium_runtime_has_rdrand(); + +# ifdef HAVE_SAFE_ARC4RANDOM + errno = errno_save; +# else + +# ifdef HAVE_LINUX_COMPATIBLE_GETRANDOM + { + unsigned char fodder[16]; + + if (randombytes_linux_getrandom(fodder, sizeof fodder) == 0) { + global.getrandom_available = 1; + errno = errno_save; + return; + } + global.getrandom_available = 0; + } +# endif /* HAVE_LINUX_COMPATIBLE_GETRANDOM */ + + if ((global.random_data_source_fd = + randombytes_salsa20_random_random_dev_open()) == -1) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } + errno = errno_save; +# endif /* HAVE_SAFE_ARC4RANDOM */ +} + +#endif /* _WIN32 */ + +/* + * (Re)seed the generator using the entropy source + */ + +static void +randombytes_salsa20_random_stir(void) +{ + stream.nonce = sodium_hrtime(); + assert(stream.nonce != (uint64_t) 0U); + memset(stream.rnd32, 0, sizeof stream.rnd32); + stream.rnd32_outleft = (size_t) 0U; + if (global.initialized == 0) { + randombytes_salsa20_random_init(); + global.initialized = 1; + } +#ifdef HAVE_GETPID + global.pid = getpid(); +#endif + +#ifndef _WIN32 + +# ifdef HAVE_SAFE_ARC4RANDOM + arc4random_buf(stream.key, sizeof stream.key); +# elif defined(HAVE_LINUX_COMPATIBLE_GETRANDOM) + if (global.getrandom_available != 0) { + if (randombytes_linux_getrandom(stream.key, sizeof stream.key) != 0) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } + } else if (global.random_data_source_fd == -1 || + safe_read(global.random_data_source_fd, stream.key, + sizeof stream.key) != (ssize_t) sizeof stream.key) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } +# else + if (global.random_data_source_fd == -1 || + safe_read(global.random_data_source_fd, stream.key, + sizeof stream.key) != (ssize_t) sizeof stream.key) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } +# endif + +#else /* _WIN32 */ +#if !defined(_XBOX_ONE) && !defined(_GAMING_XBOX) + if (! RtlGenRandom((PVOID) stream.key, (ULONG) sizeof stream.key)) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } +#endif +#endif + + stream.initialized = 1; +} + +/* + * Reseed the generator if it hasn't been initialized yet + */ + +static void +randombytes_salsa20_random_stir_if_needed(void) +{ +#ifdef HAVE_GETPID + if (stream.initialized == 0) { + randombytes_salsa20_random_stir(); + } else if (global.pid != getpid()) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } +#else + if (stream.initialized == 0) { + randombytes_salsa20_random_stir(); + } +#endif +} + +/* + * Close the stream, free global resources + */ + +#ifdef _WIN32 +static int +randombytes_salsa20_random_close(void) +{ + int ret = -1; + + if (global.initialized != 0) { + global.initialized = 0; + ret = 0; + } + sodium_memzero(&stream, sizeof stream); + + return ret; +} +#else +static int +randombytes_salsa20_random_close(void) +{ + int ret = -1; + + if (global.random_data_source_fd != -1 && + close(global.random_data_source_fd) == 0) { + global.random_data_source_fd = -1; + global.initialized = 0; +# ifdef HAVE_GETPID + global.pid = (pid_t) 0; +# endif + ret = 0; + } + +# ifdef HAVE_SAFE_ARC4RANDOM + ret = 0; +# endif + +# ifdef HAVE_LINUX_COMPATIBLE_GETRANDOM + if (global.getrandom_available != 0) { + ret = 0; + } +# endif + + sodium_memzero(&stream, sizeof stream); + + return ret; +} +#endif + +/* + * RDRAND is only used to mitigate prediction if a key is compromised + */ + +static void +randombytes_salsa20_random_xorhwrand(void) +{ +/* LCOV_EXCL_START */ +#ifdef HAVE_RDRAND + unsigned int r; + + if (global.rdrand_available == 0) { + return; + } + (void) _rdrand32_step(&r); + * (uint32_t *) (void *) + &stream.key[crypto_stream_salsa20_KEYBYTES - 4] ^= (uint32_t) r; +#endif +/* LCOV_EXCL_STOP */ +} + +/* + * XOR the key with another same-length secret + */ + +static inline void +randombytes_salsa20_random_xorkey(const unsigned char * const mix) +{ + unsigned char *key = stream.key; + size_t i; + + for (i = (size_t) 0U; i < sizeof stream.key; i++) { + key[i] ^= mix[i]; + } +} + +/* + * Put `size` random bytes into `buf` and overwrite the key + */ + +static void +randombytes_salsa20_random_buf(void * const buf, const size_t size) +{ + size_t i; + int ret; + (void) ret; + + randombytes_salsa20_random_stir_if_needed(); + COMPILER_ASSERT(sizeof stream.nonce == crypto_stream_salsa20_NONCEBYTES); +#if defined(ULLONG_MAX) && defined(SIZE_MAX) +# if SIZE_MAX > ULLONG_MAX + /* coverity[result_independent_of_operands] */ + assert(size <= ULLONG_MAX); +# endif +#endif + ret = crypto_stream_salsa20((unsigned char *) buf, (unsigned long long) size, + (unsigned char *) &stream.nonce, stream.key); + assert(ret == 0); + for (i = 0U; i < sizeof size; i++) { + stream.key[i] ^= ((const unsigned char *) (const void *) &size)[i]; + } + randombytes_salsa20_random_xorhwrand(); + stream.nonce++; + crypto_stream_salsa20_xor(stream.key, stream.key, sizeof stream.key, + (unsigned char *) &stream.nonce, stream.key); +} + +/* + * Pop a 32-bit value from the random pool + * + * Overwrite the key after the pool gets refilled. + */ + +static uint32_t +randombytes_salsa20_random(void) +{ + uint32_t val; + int ret; + (void) ret; + + COMPILER_ASSERT(sizeof stream.rnd32 >= (sizeof stream.key) + (sizeof val)); + COMPILER_ASSERT(((sizeof stream.rnd32) - (sizeof stream.key)) + % sizeof val == (size_t) 0U); + if (stream.rnd32_outleft <= (size_t) 0U) { + randombytes_salsa20_random_stir_if_needed(); + COMPILER_ASSERT(sizeof stream.nonce == crypto_stream_salsa20_NONCEBYTES); + ret = crypto_stream_salsa20((unsigned char *) stream.rnd32, + (unsigned long long) sizeof stream.rnd32, + (unsigned char *) &stream.nonce, + stream.key); + assert(ret == 0); + stream.rnd32_outleft = (sizeof stream.rnd32) - (sizeof stream.key); + randombytes_salsa20_random_xorhwrand(); + randombytes_salsa20_random_xorkey(&stream.rnd32[stream.rnd32_outleft]); + memset(&stream.rnd32[stream.rnd32_outleft], 0, sizeof stream.key); + stream.nonce++; + } + stream.rnd32_outleft -= sizeof val; + memcpy(&val, &stream.rnd32[stream.rnd32_outleft], sizeof val); + memset(&stream.rnd32[stream.rnd32_outleft], 0, sizeof val); + + return val; +} + +static const char * +randombytes_salsa20_implementation_name(void) +{ + return "salsa20"; +} + +struct randombytes_implementation randombytes_salsa20_implementation = { + SODIUM_C99(.implementation_name =) randombytes_salsa20_implementation_name, + SODIUM_C99(.random =) randombytes_salsa20_random, + SODIUM_C99(.stir =) randombytes_salsa20_random_stir, + SODIUM_C99(.uniform =) NULL, + SODIUM_C99(.buf =) randombytes_salsa20_random_buf, + SODIUM_C99(.close =) randombytes_salsa20_random_close +}; diff --git a/windows/sodium/randombytes_salsa20_random.h b/sodium/sodium_randombytes_salsa20_random.h similarity index 79% rename from windows/sodium/randombytes_salsa20_random.h rename to sodium/sodium_randombytes_salsa20_random.h index 4deae15b..f636f160 100644 --- a/windows/sodium/randombytes_salsa20_random.h +++ b/sodium/sodium_randombytes_salsa20_random.h @@ -2,14 +2,13 @@ #ifndef randombytes_salsa20_random_H #define randombytes_salsa20_random_H -#include "export.h" -#include "randombytes.h" +#include "sodium_export.h" +#include "sodium_randombytes.h" #ifdef __cplusplus extern "C" { #endif -SODIUM_EXPORT extern struct randombytes_implementation randombytes_salsa20_implementation; #ifdef __cplusplus diff --git a/sodium/sodium_randombytes_sysrandom.c b/sodium/sodium_randombytes_sysrandom.c new file mode 100644 index 00000000..0b5c3a11 --- /dev/null +++ b/sodium/sodium_randombytes_sysrandom.c @@ -0,0 +1,422 @@ + +#include +#include +#include +#include +#include +#include +#ifndef _WIN32 +# include +#endif + +#include +#include +#ifndef _WIN32 +# include +# include +#endif +#ifdef __linux__ +# ifdef __dietlibc__ +# define _LINUX_SOURCE +# include +# define HAVE_LINUX_COMPATIBLE_GETRANDOM +# else /* __dietlibc__ */ +# include +# if defined(SYS_getrandom) && defined(__NR_getrandom) +# define getrandom(B, S, F) syscall(SYS_getrandom, (B), (int) (S), (F)) +# define HAVE_LINUX_COMPATIBLE_GETRANDOM +# endif +# endif /* __dietlibc */ +#elif defined(__FreeBSD__) +# include +# if defined(__FreeBSD_version) && __FreeBSD_version >= 1200000 +# include +# define HAVE_LINUX_COMPATIBLE_GETRANDOM +# endif +#endif +#if !defined(NO_BLOCKING_RANDOM_POLL) && defined(__linux__) +# define BLOCK_ON_DEV_RANDOM +#endif +#ifdef BLOCK_ON_DEV_RANDOM +# include +#endif + +#include "sodium_core.h" +#include "sodium_private_common.h" +#include "sodium_randombytes.h" +#include "sodium_randombytes_sysrandom.h" +#include "sodium_utils.h" + +#ifdef _WIN32 +/* `RtlGenRandom` is used over `CryptGenRandom` on Microsoft Windows based systems: + * - `CryptGenRandom` requires pulling in `CryptoAPI` which causes unnecessary + * memory overhead if this API is not being used for other purposes + * - `RtlGenRandom` is thus called directly instead. A detailed explanation + * can be found here: https://blogs.msdn.microsoft.com/michael_howard/2005/01/14/cryptographically-secure-random-number-on-windows-without-using-cryptoapi/ + * + * In spite of the disclaimer on the `RtlGenRandom` documentation page that was + * written back in the Windows XP days, this function is here to stay. The CRT + * function `rand_s()` directly depends on it, so touching it would break many + * applications released since Windows XP. + * + * Also note that Rust, Firefox and BoringSSL (thus, Google Chrome and everything + * based on Chromium) also depend on it, and that libsodium allows the RNG to be + * replaced without patching nor recompiling the library. + */ +# include +# define RtlGenRandom SystemFunction036 +# if defined(__cplusplus) +extern "C" +# endif +# if !defined(_XBOX_ONE) && !defined(_GAMING_XBOX) +BOOLEAN NTAPI RtlGenRandom(PVOID RandomBuffer, ULONG RandomBufferLength); +# pragma comment(lib, "advapi32.lib") +# endif +#endif + +#if defined(__OpenBSD__) || defined(__CloudABI__) +# define HAVE_SAFE_ARC4RANDOM 1 +#endif + +#ifndef SSIZE_MAX +# define SSIZE_MAX (SIZE_MAX / 2 - 1) +#endif + +#ifdef HAVE_SAFE_ARC4RANDOM + +static uint32_t +randombytes_sysrandom(void) +{ + return arc4random(); +} + +static void +randombytes_sysrandom_stir(void) +{ +} + +static void +randombytes_sysrandom_buf(void * const buf, const size_t size) +{ + arc4random_buf(buf, size); +} + +static int +randombytes_sysrandom_close(void) +{ + return 0; +} + +#elif defined(__ORBIS__) || defined(__PROSPERO__) + +static uint32_t +randombytes_sysrandom(void) +{ + return 0; +} + +static void +randombytes_sysrandom_stir(void) +{ +} + +static void +randombytes_sysrandom_buf(void * const buf, const size_t size) +{ +} + +static int +randombytes_sysrandom_close(void) +{ + return 0; +} + +#else /* __OpenBSD__ */ + +typedef struct SysRandom_ { + int random_data_source_fd; + int initialized; + int getrandom_available; +} SysRandom; + +static SysRandom stream = { + SODIUM_C99(.random_data_source_fd =) -1, + SODIUM_C99(.initialized =) 0, + SODIUM_C99(.getrandom_available =) 0 +}; + +# ifndef _WIN32 +static ssize_t +safe_read(const int fd, void * const buf_, size_t size) +{ + unsigned char *buf = (unsigned char *) buf_; + ssize_t readnb; + + assert(size > (size_t) 0U); + assert(size <= SSIZE_MAX); + do { + while ((readnb = read(fd, buf, size)) < (ssize_t) 0 && + (errno == EINTR || errno == EAGAIN)); /* LCOV_EXCL_LINE */ + if (readnb < (ssize_t) 0) { + return readnb; /* LCOV_EXCL_LINE */ + } + if (readnb == (ssize_t) 0) { + break; /* LCOV_EXCL_LINE */ + } + size -= (size_t) readnb; + buf += readnb; + } while (size > (ssize_t) 0); + + return (ssize_t) (buf - (unsigned char *) buf_); +} + +# ifdef BLOCK_ON_DEV_RANDOM +static int +randombytes_block_on_dev_random(void) +{ + struct pollfd pfd; + int fd; + int pret; + + fd = open("/dev/random", O_RDONLY); + if (fd == -1) { + return 0; + } + pfd.fd = fd; + pfd.events = POLLIN; + pfd.revents = 0; + do { + pret = poll(&pfd, 1, -1); + } while (pret < 0 && (errno == EINTR || errno == EAGAIN)); + if (pret != 1) { + (void) close(fd); + errno = EIO; + return -1; + } + return close(fd); +} +# endif /* BLOCK_ON_DEV_RANDOM */ + +static int +randombytes_sysrandom_random_dev_open(void) +{ +/* LCOV_EXCL_START */ + struct stat st; + static const char *devices[] = { +# ifndef USE_BLOCKING_RANDOM + "/dev/urandom", +# endif + "/dev/random", NULL + }; + const char **device = devices; + int fd; + +# ifdef BLOCK_ON_DEV_RANDOM + if (randombytes_block_on_dev_random() != 0) { + return -1; + } +# endif + do { + fd = open(*device, O_RDONLY); + if (fd != -1) { + if (fstat(fd, &st) == 0 && +# ifdef __COMPCERT__ + 1 +# elif defined(S_ISNAM) + (S_ISNAM(st.st_mode) || S_ISCHR(st.st_mode)) +# else + S_ISCHR(st.st_mode) +# endif + ) { +# if defined(F_SETFD) && defined(FD_CLOEXEC) && !defined(NN_NINTENDO_SDK) + (void) fcntl(fd, F_SETFD, fcntl(fd, F_GETFD) | FD_CLOEXEC); +# endif + return fd; + } + (void) close(fd); + } else if (errno == EINTR) { + continue; + } + device++; + } while (*device != NULL); + + errno = EIO; + return -1; +/* LCOV_EXCL_STOP */ +} + +# ifdef HAVE_LINUX_COMPATIBLE_GETRANDOM +static int +_randombytes_linux_getrandom(void * const buf, const size_t size) +{ + int readnb; + + assert(size <= 256U); + do { + readnb = getrandom(buf, size, 0); + } while (readnb < 0 && (errno == EINTR || errno == EAGAIN)); + + return (readnb == (int) size) - 1; +} + +static int +randombytes_linux_getrandom(void * const buf_, size_t size) +{ + unsigned char *buf = (unsigned char *) buf_; + size_t chunk_size = 256U; + + do { + if (size < chunk_size) { + chunk_size = size; + assert(chunk_size > (size_t) 0U); + } + if (_randombytes_linux_getrandom(buf, chunk_size) != 0) { + return -1; + } + size -= chunk_size; + buf += chunk_size; + } while (size > (size_t) 0U); + + return 0; +} +# endif /* HAVE_LINUX_COMPATIBLE_GETRANDOM */ + +static void +randombytes_sysrandom_init(void) +{ + const int errno_save = errno; + +# ifdef HAVE_LINUX_COMPATIBLE_GETRANDOM + { + unsigned char fodder[16]; + + if (randombytes_linux_getrandom(fodder, sizeof fodder) == 0) { + stream.getrandom_available = 1; + errno = errno_save; + return; + } + stream.getrandom_available = 0; + } +# endif + + if ((stream.random_data_source_fd = + randombytes_sysrandom_random_dev_open()) == -1) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } + errno = errno_save; +} + +# else /* _WIN32 */ + +static void +randombytes_sysrandom_init(void) +{ +} +# endif /* _WIN32 */ + +static void +randombytes_sysrandom_stir(void) +{ + if (stream.initialized == 0) { + randombytes_sysrandom_init(); + stream.initialized = 1; + } +} + +static void +randombytes_sysrandom_stir_if_needed(void) +{ + if (stream.initialized == 0) { + randombytes_sysrandom_stir(); + } +} + +static int +randombytes_sysrandom_close(void) +{ + int ret = -1; + +# ifndef _WIN32 + if (stream.random_data_source_fd != -1 && + close(stream.random_data_source_fd) == 0) { + stream.random_data_source_fd = -1; + stream.initialized = 0; + ret = 0; + } +# ifdef HAVE_LINUX_COMPATIBLE_GETRANDOM + if (stream.getrandom_available != 0) { + ret = 0; + } +# endif +# else /* _WIN32 */ + if (stream.initialized != 0) { + stream.initialized = 0; + ret = 0; + } +# endif /* _WIN32 */ + return ret; +} + +static void +randombytes_sysrandom_buf(void * const buf, const size_t size) +{ + (void) buf; + randombytes_sysrandom_stir_if_needed(); +# if defined(ULLONG_MAX) && defined(SIZE_MAX) +# if SIZE_MAX > ULLONG_MAX + /* coverity[result_independent_of_operands] */ + assert(size <= ULLONG_MAX); +# endif +# endif +# ifndef _WIN32 +# ifdef HAVE_LINUX_COMPATIBLE_GETRANDOM + if (stream.getrandom_available != 0) { + if (randombytes_linux_getrandom(buf, size) != 0) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } + return; + } +# endif + if (stream.random_data_source_fd == -1 || + safe_read(stream.random_data_source_fd, buf, size) != (ssize_t) size) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } +# else /* _WIN32 */ + COMPILER_ASSERT(randombytes_BYTES_MAX <= 0xffffffffUL); + if (size > (size_t) 0xffffffffUL) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } +# if !defined(_XBOX_ONE) && !defined(_GAMING_XBOX) + if (! RtlGenRandom((PVOID) buf, (ULONG) size)) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } +# endif +# endif /* _WIN32 */ +} + +static uint32_t +randombytes_sysrandom(void) +{ + uint32_t r; + + randombytes_sysrandom_buf(&r, sizeof r); + + return r; +} + +#endif /* __OpenBSD__ */ + +static const char * +randombytes_sysrandom_implementation_name(void) +{ + return "sysrandom"; +} + +struct randombytes_implementation randombytes_sysrandom_implementation = { + SODIUM_C99(.implementation_name =) randombytes_sysrandom_implementation_name, + SODIUM_C99(.random =) randombytes_sysrandom, + SODIUM_C99(.stir =) randombytes_sysrandom_stir, + SODIUM_C99(.uniform =) NULL, + SODIUM_C99(.buf =) randombytes_sysrandom_buf, + SODIUM_C99(.close =) randombytes_sysrandom_close +}; diff --git a/windows/sodium/randombytes_sysrandom.h b/sodium/sodium_randombytes_sysrandom.h similarity index 79% rename from windows/sodium/randombytes_sysrandom.h rename to sodium/sodium_randombytes_sysrandom.h index 9e27b674..c3d124f7 100644 --- a/windows/sodium/randombytes_sysrandom.h +++ b/sodium/sodium_randombytes_sysrandom.h @@ -2,14 +2,13 @@ #ifndef randombytes_sysrandom_H #define randombytes_sysrandom_H -#include "export.h" -#include "randombytes.h" +#include "sodium_export.h" +#include "sodium_randombytes.h" #ifdef __cplusplus extern "C" { #endif -SODIUM_EXPORT extern struct randombytes_implementation randombytes_sysrandom_implementation; #ifdef __cplusplus diff --git a/sodium/sodium_ref10_ed25519.c b/sodium/sodium_ref10_ed25519.c new file mode 100644 index 00000000..c9f73649 --- /dev/null +++ b/sodium/sodium_ref10_ed25519.c @@ -0,0 +1,2601 @@ +#include +#include +#include +#include + +#include "sodium_crypto_verify_32.h" +#include "sodium_private_common.h" +#include "sodium_private_ed25519_ref10.h" +#include "sodium_utils.h" + +#if defined( _MSC_VER ) +#pragma warning(disable:4127) +#pragma warning(disable:4244) +#pragma warning(disable:4668) +#pragma warning(disable:4456) +#endif + +static inline uint64_t +load_3(const unsigned char *in) +{ + uint64_t result; + + result = (uint64_t) in[0]; + result |= ((uint64_t) in[1]) << 8; + result |= ((uint64_t) in[2]) << 16; + + return result; +} + +static inline uint64_t +load_4(const unsigned char *in) +{ + uint64_t result; + + result = (uint64_t) in[0]; + result |= ((uint64_t) in[1]) << 8; + result |= ((uint64_t) in[2]) << 16; + result |= ((uint64_t) in[3]) << 24; + + return result; +} + +/* + * Field arithmetic: + * Use 5*51 bit limbs on 64-bit systems with support for 128 bit arithmetic, + * and 10*25.5 bit limbs elsewhere. + * + * Functions used elsewhere that are candidates for inlining are defined + * via "private/curve25519_ref10.h". + */ + +#ifdef HAVE_TI_MODE +# include "sodium_fe_51_constants.h" +# include "sodium_fe_51_fe.h" +#else +# include "sodium_fe_25_5_constants.h" +# include "sodium_fe_25_5_fe.h" +#endif + +void +fe25519_invert(fe25519 out, const fe25519 z) +{ + fe25519 t0; + fe25519 t1; + fe25519 t2; + fe25519 t3; + int i; + + fe25519_sq(t0, z); + fe25519_sq(t1, t0); + fe25519_sq(t1, t1); + fe25519_mul(t1, z, t1); + fe25519_mul(t0, t0, t1); + fe25519_sq(t2, t0); + fe25519_mul(t1, t1, t2); + fe25519_sq(t2, t1); + for (i = 1; i < 5; ++i) { + fe25519_sq(t2, t2); + } + fe25519_mul(t1, t2, t1); + fe25519_sq(t2, t1); + for (i = 1; i < 10; ++i) { + fe25519_sq(t2, t2); + } + fe25519_mul(t2, t2, t1); + fe25519_sq(t3, t2); + for (i = 1; i < 20; ++i) { + fe25519_sq(t3, t3); + } + fe25519_mul(t2, t3, t2); + fe25519_sq(t2, t2); + for (i = 1; i < 10; ++i) { + fe25519_sq(t2, t2); + } + fe25519_mul(t1, t2, t1); + fe25519_sq(t2, t1); + for (i = 1; i < 50; ++i) { + fe25519_sq(t2, t2); + } + fe25519_mul(t2, t2, t1); + fe25519_sq(t3, t2); + for (i = 1; i < 100; ++i) { + fe25519_sq(t3, t3); + } + fe25519_mul(t2, t3, t2); + fe25519_sq(t2, t2); + for (i = 1; i < 50; ++i) { + fe25519_sq(t2, t2); + } + fe25519_mul(t1, t2, t1); + fe25519_sq(t1, t1); + for (i = 1; i < 5; ++i) { + fe25519_sq(t1, t1); + } + fe25519_mul(out, t1, t0); +} + +static void +fe25519_pow22523(fe25519 out, const fe25519 z) +{ + fe25519 t0; + fe25519 t1; + fe25519 t2; + int i; + + fe25519_sq(t0, z); + fe25519_sq(t1, t0); + fe25519_sq(t1, t1); + fe25519_mul(t1, z, t1); + fe25519_mul(t0, t0, t1); + fe25519_sq(t0, t0); + fe25519_mul(t0, t1, t0); + fe25519_sq(t1, t0); + for (i = 1; i < 5; ++i) { + fe25519_sq(t1, t1); + } + fe25519_mul(t0, t1, t0); + fe25519_sq(t1, t0); + for (i = 1; i < 10; ++i) { + fe25519_sq(t1, t1); + } + fe25519_mul(t1, t1, t0); + fe25519_sq(t2, t1); + for (i = 1; i < 20; ++i) { + fe25519_sq(t2, t2); + } + fe25519_mul(t1, t2, t1); + fe25519_sq(t1, t1); + for (i = 1; i < 10; ++i) { + fe25519_sq(t1, t1); + } + fe25519_mul(t0, t1, t0); + fe25519_sq(t1, t0); + for (i = 1; i < 50; ++i) { + fe25519_sq(t1, t1); + } + fe25519_mul(t1, t1, t0); + fe25519_sq(t2, t1); + for (i = 1; i < 100; ++i) { + fe25519_sq(t2, t2); + } + fe25519_mul(t1, t2, t1); + fe25519_sq(t1, t1); + for (i = 1; i < 50; ++i) { + fe25519_sq(t1, t1); + } + fe25519_mul(t0, t1, t0); + fe25519_sq(t0, t0); + fe25519_sq(t0, t0); + fe25519_mul(out, t0, z); +} + +/* + r = p + q + */ + +void +ge25519_add(ge25519_p1p1 *r, const ge25519_p3 *p, const ge25519_cached *q) +{ + fe25519 t0; + + fe25519_add(r->X, p->Y, p->X); + fe25519_sub(r->Y, p->Y, p->X); + fe25519_mul(r->Z, r->X, q->YplusX); + fe25519_mul(r->Y, r->Y, q->YminusX); + fe25519_mul(r->T, q->T2d, p->T); + fe25519_mul(r->X, p->Z, q->Z); + fe25519_add(t0, r->X, r->X); + fe25519_sub(r->X, r->Z, r->Y); + fe25519_add(r->Y, r->Z, r->Y); + fe25519_add(r->Z, t0, r->T); + fe25519_sub(r->T, t0, r->T); +} + +static void +slide_vartime(signed char *r, const unsigned char *a) +{ + int i; + int b; + int k; + int ribs; + int cmp; + + for (i = 0; i < 256; ++i) { + r[i] = 1 & (a[i >> 3] >> (i & 7)); + } + for (i = 0; i < 256; ++i) { + if (! r[i]) { + continue; + } + for (b = 1; b <= 6 && i + b < 256; ++b) { + if (! r[i + b]) { + continue; + } + ribs = r[i + b] << b; + cmp = r[i] + ribs; + if (cmp <= 15) { + r[i] = cmp; + r[i + b] = 0; + } else { + cmp = r[i] - ribs; + if (cmp < -15) { + break; + } + r[i] = cmp; + for (k = i + b; k < 256; ++k) { + if (! r[k]) { + r[k] = 1; + break; + } + r[k] = 0; + } + } + } + } +} + +int +ge25519_frombytes(ge25519_p3 *h, const unsigned char *s) +{ + fe25519 u; + fe25519 v; + fe25519 v3; + fe25519 vxx; + fe25519 m_root_check, p_root_check; + fe25519 negx; + fe25519 x_sqrtm1; + int has_m_root, has_p_root; + + fe25519_frombytes(h->Y, s); + fe25519_1(h->Z); + fe25519_sq(u, h->Y); + fe25519_mul(v, u, d); + fe25519_sub(u, u, h->Z); /* u = y^2-1 */ + fe25519_add(v, v, h->Z); /* v = dy^2+1 */ + + fe25519_sq(v3, v); + fe25519_mul(v3, v3, v); /* v3 = v^3 */ + fe25519_sq(h->X, v3); + fe25519_mul(h->X, h->X, v); + fe25519_mul(h->X, h->X, u); /* x = uv^7 */ + + fe25519_pow22523(h->X, h->X); /* x = (uv^7)^((q-5)/8) */ + fe25519_mul(h->X, h->X, v3); + fe25519_mul(h->X, h->X, u); /* x = uv^3(uv^7)^((q-5)/8) */ + + fe25519_sq(vxx, h->X); + fe25519_mul(vxx, vxx, v); + fe25519_sub(m_root_check, vxx, u); /* vx^2-u */ + fe25519_add(p_root_check, vxx, u); /* vx^2+u */ + has_m_root = fe25519_iszero(m_root_check); + has_p_root = fe25519_iszero(p_root_check); + fe25519_mul(x_sqrtm1, h->X, sqrtm1); /* x*sqrt(-1) */ + fe25519_cmov(h->X, x_sqrtm1, 1 - has_m_root); + + fe25519_neg(negx, h->X); + fe25519_cmov(h->X, negx, fe25519_isnegative(h->X) ^ (s[31] >> 7)); + fe25519_mul(h->T, h->X, h->Y); + + return (has_m_root | has_p_root) - 1; +} + +int +ge25519_frombytes_negate_vartime(ge25519_p3 *h, const unsigned char *s) +{ + fe25519 u; + fe25519 v; + fe25519 v3; + fe25519 vxx; + fe25519 m_root_check, p_root_check; + + fe25519_frombytes(h->Y, s); + fe25519_1(h->Z); + fe25519_sq(u, h->Y); + fe25519_mul(v, u, d); + fe25519_sub(u, u, h->Z); /* u = y^2-1 */ + fe25519_add(v, v, h->Z); /* v = dy^2+1 */ + + fe25519_sq(v3, v); + fe25519_mul(v3, v3, v); /* v3 = v^3 */ + fe25519_sq(h->X, v3); + fe25519_mul(h->X, h->X, v); + fe25519_mul(h->X, h->X, u); /* x = uv^7 */ + + fe25519_pow22523(h->X, h->X); /* x = (uv^7)^((q-5)/8) */ + fe25519_mul(h->X, h->X, v3); + fe25519_mul(h->X, h->X, u); /* x = uv^3(uv^7)^((q-5)/8) */ + + fe25519_sq(vxx, h->X); + fe25519_mul(vxx, vxx, v); + fe25519_sub(m_root_check, vxx, u); /* vx^2-u */ + if (fe25519_iszero(m_root_check) == 0) { + fe25519_add(p_root_check, vxx, u); /* vx^2+u */ + if (fe25519_iszero(p_root_check) == 0) { + return -1; + } + fe25519_mul(h->X, h->X, sqrtm1); + } + + if (fe25519_isnegative(h->X) == (s[31] >> 7)) { + fe25519_neg(h->X, h->X); + } + fe25519_mul(h->T, h->X, h->Y); + + return 0; +} + +/* + r = p + q + */ + +static void +ge25519_madd(ge25519_p1p1 *r, const ge25519_p3 *p, const ge25519_precomp *q) +{ + fe25519 t0; + + fe25519_add(r->X, p->Y, p->X); + fe25519_sub(r->Y, p->Y, p->X); + fe25519_mul(r->Z, r->X, q->yplusx); + fe25519_mul(r->Y, r->Y, q->yminusx); + fe25519_mul(r->T, q->xy2d, p->T); + fe25519_add(t0, p->Z, p->Z); + fe25519_sub(r->X, r->Z, r->Y); + fe25519_add(r->Y, r->Z, r->Y); + fe25519_add(r->Z, t0, r->T); + fe25519_sub(r->T, t0, r->T); +} + +/* + r = p - q + */ + +static void +ge25519_msub(ge25519_p1p1 *r, const ge25519_p3 *p, const ge25519_precomp *q) +{ + fe25519 t0; + + fe25519_add(r->X, p->Y, p->X); + fe25519_sub(r->Y, p->Y, p->X); + fe25519_mul(r->Z, r->X, q->yminusx); + fe25519_mul(r->Y, r->Y, q->yplusx); + fe25519_mul(r->T, q->xy2d, p->T); + fe25519_add(t0, p->Z, p->Z); + fe25519_sub(r->X, r->Z, r->Y); + fe25519_add(r->Y, r->Z, r->Y); + fe25519_sub(r->Z, t0, r->T); + fe25519_add(r->T, t0, r->T); +} + +/* + r = p + */ + +void +ge25519_p1p1_to_p2(ge25519_p2 *r, const ge25519_p1p1 *p) +{ + fe25519_mul(r->X, p->X, p->T); + fe25519_mul(r->Y, p->Y, p->Z); + fe25519_mul(r->Z, p->Z, p->T); +} + +/* + r = p + */ + +void +ge25519_p1p1_to_p3(ge25519_p3 *r, const ge25519_p1p1 *p) +{ + fe25519_mul(r->X, p->X, p->T); + fe25519_mul(r->Y, p->Y, p->Z); + fe25519_mul(r->Z, p->Z, p->T); + fe25519_mul(r->T, p->X, p->Y); +} + +static void +ge25519_p2_0(ge25519_p2 *h) +{ + fe25519_0(h->X); + fe25519_1(h->Y); + fe25519_1(h->Z); +} + +/* + r = 2 * p + */ + +static void +ge25519_p2_dbl(ge25519_p1p1 *r, const ge25519_p2 *p) +{ + fe25519 t0; + + fe25519_sq(r->X, p->X); + fe25519_sq(r->Z, p->Y); + fe25519_sq2(r->T, p->Z); + fe25519_add(r->Y, p->X, p->Y); + fe25519_sq(t0, r->Y); + fe25519_add(r->Y, r->Z, r->X); + fe25519_sub(r->Z, r->Z, r->X); + fe25519_sub(r->X, t0, r->Y); + fe25519_sub(r->T, r->T, r->Z); +} + +static void +ge25519_p3_0(ge25519_p3 *h) +{ + fe25519_0(h->X); + fe25519_1(h->Y); + fe25519_1(h->Z); + fe25519_0(h->T); +} + +static void +ge25519_cached_0(ge25519_cached *h) +{ + fe25519_1(h->YplusX); + fe25519_1(h->YminusX); + fe25519_1(h->Z); + fe25519_0(h->T2d); +} + +/* + r = p + */ + +void +ge25519_p3_to_cached(ge25519_cached *r, const ge25519_p3 *p) +{ + fe25519_add(r->YplusX, p->Y, p->X); + fe25519_sub(r->YminusX, p->Y, p->X); + fe25519_copy(r->Z, p->Z); + fe25519_mul(r->T2d, p->T, d2); +} + +#if defined(__ORBIS__) || defined(__PROSPERO__) || defined(__APPLE__) || defined(__ANDROID__) +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wunused-function" +#endif // #if defined(__ORBIS__) || defined(__PROSPERO__) || defined(__APPLE__) || defined(__ANDROID__) + +static void +ge25519_p3_to_precomp(ge25519_precomp *pi, const ge25519_p3 *p) +{ + fe25519 recip; + fe25519 x; + fe25519 y; + fe25519 xy; + + fe25519_invert(recip, p->Z); + fe25519_mul(x, p->X, recip); + fe25519_mul(y, p->Y, recip); + fe25519_add(pi->yplusx, y, x); + fe25519_sub(pi->yminusx, y, x); + fe25519_mul(xy, x, y); + fe25519_mul(pi->xy2d, xy, d2); +} + +#if defined(__ORBIS__) || defined(__PROSPERO__) || defined(__APPLE__) || defined(__ANDROID__) +#pragma clang diagnostic pop +#endif // #if defined(__ORBIS__) || defined(__PROSPERO__) || defined(__APPLE__) || defined(__ANDROID__) + +/* + r = p + */ + +static void +ge25519_p3_to_p2(ge25519_p2 *r, const ge25519_p3 *p) +{ + fe25519_copy(r->X, p->X); + fe25519_copy(r->Y, p->Y); + fe25519_copy(r->Z, p->Z); +} + +void +ge25519_p3_tobytes(unsigned char *s, const ge25519_p3 *h) +{ + fe25519 recip; + fe25519 x; + fe25519 y; + + fe25519_invert(recip, h->Z); + fe25519_mul(x, h->X, recip); + fe25519_mul(y, h->Y, recip); + fe25519_tobytes(s, y); + s[31] ^= fe25519_isnegative(x) << 7; +} + +/* + r = 2 * p + */ + +static void +ge25519_p3_dbl(ge25519_p1p1 *r, const ge25519_p3 *p) +{ + ge25519_p2 q; + ge25519_p3_to_p2(&q, p); + ge25519_p2_dbl(r, &q); +} + +static void +ge25519_precomp_0(ge25519_precomp *h) +{ + fe25519_1(h->yplusx); + fe25519_1(h->yminusx); + fe25519_0(h->xy2d); +} + +static unsigned char +equal(signed char b, signed char c) +{ + unsigned char ub = b; + unsigned char uc = c; + unsigned char x = ub ^ uc; /* 0: yes; 1..255: no */ + uint32_t y = x; /* 0: yes; 1..255: no */ + + y -= 1; /* 4294967295: yes; 0..254: no */ + y >>= 31; /* 1: yes; 0: no */ + + return y; +} + +static unsigned char +negative(signed char b) +{ + /* 18446744073709551361..18446744073709551615: yes; 0..255: no */ + uint64_t x = b; + + x >>= 63; /* 1: yes; 0: no */ + + return x; +} + +static void +ge25519_cmov(ge25519_precomp *t, const ge25519_precomp *u, unsigned char b) +{ + fe25519_cmov(t->yplusx, u->yplusx, b); + fe25519_cmov(t->yminusx, u->yminusx, b); + fe25519_cmov(t->xy2d, u->xy2d, b); +} + +static void +ge25519_cmov_cached(ge25519_cached *t, const ge25519_cached *u, unsigned char b) +{ + fe25519_cmov(t->YplusX, u->YplusX, b); + fe25519_cmov(t->YminusX, u->YminusX, b); + fe25519_cmov(t->Z, u->Z, b); + fe25519_cmov(t->T2d, u->T2d, b); +} + +static void +ge25519_select(ge25519_precomp *t, const ge25519_precomp precomp[8], const signed char b) +{ + ge25519_precomp minust; + const unsigned char bnegative = negative(b); + const unsigned char babs = b - (((-bnegative) & b) * ((signed char) 1 << 1)); + + ge25519_precomp_0(t); + ge25519_cmov(t, &precomp[0], equal(babs, 1)); + ge25519_cmov(t, &precomp[1], equal(babs, 2)); + ge25519_cmov(t, &precomp[2], equal(babs, 3)); + ge25519_cmov(t, &precomp[3], equal(babs, 4)); + ge25519_cmov(t, &precomp[4], equal(babs, 5)); + ge25519_cmov(t, &precomp[5], equal(babs, 6)); + ge25519_cmov(t, &precomp[6], equal(babs, 7)); + ge25519_cmov(t, &precomp[7], equal(babs, 8)); + fe25519_copy(minust.yplusx, t->yminusx); + fe25519_copy(minust.yminusx, t->yplusx); + fe25519_neg(minust.xy2d, t->xy2d); + ge25519_cmov(t, &minust, bnegative); +} + +static void +ge25519_select_base(ge25519_precomp *t, const int pos, const signed char b) +{ + static const ge25519_precomp base[32][8] = { /* base[i][j] = (j+1)*256^i*B */ +#ifdef HAVE_TI_MODE +# include "sodium_fe_51_base.h" +#else +# include "sodium_fe_25_5_base.h" +#endif + }; + ge25519_select(t, base[pos], b); +} + +static void +ge25519_select_cached(ge25519_cached *t, const ge25519_cached cached[8], const signed char b) +{ + ge25519_cached minust; + const unsigned char bnegative = negative(b); + const unsigned char babs = b - (((-bnegative) & b) * ((signed char) 1 << 1)); + + ge25519_cached_0(t); + ge25519_cmov_cached(t, &cached[0], equal(babs, 1)); + ge25519_cmov_cached(t, &cached[1], equal(babs, 2)); + ge25519_cmov_cached(t, &cached[2], equal(babs, 3)); + ge25519_cmov_cached(t, &cached[3], equal(babs, 4)); + ge25519_cmov_cached(t, &cached[4], equal(babs, 5)); + ge25519_cmov_cached(t, &cached[5], equal(babs, 6)); + ge25519_cmov_cached(t, &cached[6], equal(babs, 7)); + ge25519_cmov_cached(t, &cached[7], equal(babs, 8)); + fe25519_copy(minust.YplusX, t->YminusX); + fe25519_copy(minust.YminusX, t->YplusX); + fe25519_copy(minust.Z, t->Z); + fe25519_neg(minust.T2d, t->T2d); + ge25519_cmov_cached(t, &minust, bnegative); +} + +/* + r = p - q + */ + +void +ge25519_sub(ge25519_p1p1 *r, const ge25519_p3 *p, const ge25519_cached *q) +{ + fe25519 t0; + + fe25519_add(r->X, p->Y, p->X); + fe25519_sub(r->Y, p->Y, p->X); + fe25519_mul(r->Z, r->X, q->YminusX); + fe25519_mul(r->Y, r->Y, q->YplusX); + fe25519_mul(r->T, q->T2d, p->T); + fe25519_mul(r->X, p->Z, q->Z); + fe25519_add(t0, r->X, r->X); + fe25519_sub(r->X, r->Z, r->Y); + fe25519_add(r->Y, r->Z, r->Y); + fe25519_sub(r->Z, t0, r->T); + fe25519_add(r->T, t0, r->T); +} + +void +ge25519_tobytes(unsigned char *s, const ge25519_p2 *h) +{ + fe25519 recip; + fe25519 x; + fe25519 y; + + fe25519_invert(recip, h->Z); + fe25519_mul(x, h->X, recip); + fe25519_mul(y, h->Y, recip); + fe25519_tobytes(s, y); + s[31] ^= fe25519_isnegative(x) << 7; +} + +/* + r = a * A + b * B + where a = a[0]+256*a[1]+...+256^31 a[31]. + and b = b[0]+256*b[1]+...+256^31 b[31]. + B is the Ed25519 base point (x,4/5) with x positive. + + Only used for signatures verification. + */ + +void +ge25519_double_scalarmult_vartime(ge25519_p2 *r, const unsigned char *a, + const ge25519_p3 *A, const unsigned char *b) +{ + static const ge25519_precomp Bi[8] = { +#ifdef HAVE_TI_MODE +# include "sodium_fe_51_base2.h" +#else +# include "sodium_fe_25_5_base2.h" +#endif + }; + signed char aslide[256]; + signed char bslide[256]; + ge25519_cached Ai[8]; /* A,3A,5A,7A,9A,11A,13A,15A */ + ge25519_p1p1 t; + ge25519_p3 u; + ge25519_p3 A2; + int i; + + slide_vartime(aslide, a); + slide_vartime(bslide, b); + + ge25519_p3_to_cached(&Ai[0], A); + + ge25519_p3_dbl(&t, A); + ge25519_p1p1_to_p3(&A2, &t); + + ge25519_add(&t, &A2, &Ai[0]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[1], &u); + + ge25519_add(&t, &A2, &Ai[1]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[2], &u); + + ge25519_add(&t, &A2, &Ai[2]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[3], &u); + + ge25519_add(&t, &A2, &Ai[3]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[4], &u); + + ge25519_add(&t, &A2, &Ai[4]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[5], &u); + + ge25519_add(&t, &A2, &Ai[5]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[6], &u); + + ge25519_add(&t, &A2, &Ai[6]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[7], &u); + + ge25519_p2_0(r); + + for (i = 255; i >= 0; --i) { + if (aslide[i] || bslide[i]) { + break; + } + } + + for (; i >= 0; --i) { + ge25519_p2_dbl(&t, r); + + if (aslide[i] > 0) { + ge25519_p1p1_to_p3(&u, &t); + ge25519_add(&t, &u, &Ai[aslide[i] / 2]); + } else if (aslide[i] < 0) { + ge25519_p1p1_to_p3(&u, &t); + ge25519_sub(&t, &u, &Ai[(-aslide[i]) / 2]); + } + + if (bslide[i] > 0) { + ge25519_p1p1_to_p3(&u, &t); + ge25519_madd(&t, &u, &Bi[bslide[i] / 2]); + } else if (bslide[i] < 0) { + ge25519_p1p1_to_p3(&u, &t); + ge25519_msub(&t, &u, &Bi[(-bslide[i]) / 2]); + } + + ge25519_p1p1_to_p2(r, &t); + } +} + +/* + h = a * p + where a = a[0]+256*a[1]+...+256^31 a[31] + + Preconditions: + a[31] <= 127 + + p is public + */ + +void +ge25519_scalarmult(ge25519_p3 *h, const unsigned char *a, const ge25519_p3 *p) +{ + signed char e[64]; + signed char carry; + ge25519_p1p1 r; + ge25519_p2 s; + ge25519_p1p1 t2, t3, t4, t5, t6, t7, t8; + ge25519_p3 p2, p3, p4, p5, p6, p7, p8; + ge25519_cached pi[8]; + ge25519_cached t; + int i; + + ge25519_p3_to_cached(&pi[1 - 1], p); /* p */ + + ge25519_p3_dbl(&t2, p); + ge25519_p1p1_to_p3(&p2, &t2); + ge25519_p3_to_cached(&pi[2 - 1], &p2); /* 2p = 2*p */ + + ge25519_add(&t3, p, &pi[2 - 1]); + ge25519_p1p1_to_p3(&p3, &t3); + ge25519_p3_to_cached(&pi[3 - 1], &p3); /* 3p = 2p+p */ + + ge25519_p3_dbl(&t4, &p2); + ge25519_p1p1_to_p3(&p4, &t4); + ge25519_p3_to_cached(&pi[4 - 1], &p4); /* 4p = 2*2p */ + + ge25519_add(&t5, p, &pi[4 - 1]); + ge25519_p1p1_to_p3(&p5, &t5); + ge25519_p3_to_cached(&pi[5 - 1], &p5); /* 5p = 4p+p */ + + ge25519_p3_dbl(&t6, &p3); + ge25519_p1p1_to_p3(&p6, &t6); + ge25519_p3_to_cached(&pi[6 - 1], &p6); /* 6p = 2*3p */ + + ge25519_add(&t7, p, &pi[6 - 1]); + ge25519_p1p1_to_p3(&p7, &t7); + ge25519_p3_to_cached(&pi[7 - 1], &p7); /* 7p = 6p+p */ + + ge25519_p3_dbl(&t8, &p4); + ge25519_p1p1_to_p3(&p8, &t8); + ge25519_p3_to_cached(&pi[8 - 1], &p8); /* 8p = 2*4p */ + + for (i = 0; i < 32; ++i) { + e[2 * i + 0] = (a[i] >> 0) & 15; + e[2 * i + 1] = (a[i] >> 4) & 15; + } + /* each e[i] is between 0 and 15 */ + /* e[63] is between 0 and 7 */ + + carry = 0; + for (i = 0; i < 63; ++i) { + e[i] += carry; + carry = e[i] + 8; + carry >>= 4; + e[i] -= carry * ((signed char) 1 << 4); + } + e[63] += carry; + /* each e[i] is between -8 and 8 */ + + ge25519_p3_0(h); + + for (i = 63; i != 0; i--) { + ge25519_select_cached(&t, pi, e[i]); + ge25519_add(&r, h, &t); + + ge25519_p1p1_to_p2(&s, &r); + ge25519_p2_dbl(&r, &s); + ge25519_p1p1_to_p2(&s, &r); + ge25519_p2_dbl(&r, &s); + ge25519_p1p1_to_p2(&s, &r); + ge25519_p2_dbl(&r, &s); + ge25519_p1p1_to_p2(&s, &r); + ge25519_p2_dbl(&r, &s); + + ge25519_p1p1_to_p3(h, &r); /* *16 */ + } + ge25519_select_cached(&t, pi, e[i]); + ge25519_add(&r, h, &t); + + ge25519_p1p1_to_p3(h, &r); +} + +/* + h = a * B (with precomputation) + where a = a[0]+256*a[1]+...+256^31 a[31] + B is the Ed25519 base point (x,4/5) with x positive + (as bytes: 0x5866666666666666666666666666666666666666666666666666666666666666) + + Preconditions: + a[31] <= 127 + */ + +void +ge25519_scalarmult_base(ge25519_p3 *h, const unsigned char *a) +{ + signed char e[64]; + signed char carry; + ge25519_p1p1 r; + ge25519_p2 s; + ge25519_precomp t; + int i; + + for (i = 0; i < 32; ++i) { + e[2 * i + 0] = (a[i] >> 0) & 15; + e[2 * i + 1] = (a[i] >> 4) & 15; + } + /* each e[i] is between 0 and 15 */ + /* e[63] is between 0 and 7 */ + + carry = 0; + for (i = 0; i < 63; ++i) { + e[i] += carry; + carry = e[i] + 8; + carry >>= 4; + e[i] -= carry * ((signed char) 1 << 4); + } + e[63] += carry; + /* each e[i] is between -8 and 8 */ + + ge25519_p3_0(h); + + for (i = 1; i < 64; i += 2) { + ge25519_select_base(&t, i / 2, e[i]); + ge25519_madd(&r, h, &t); + ge25519_p1p1_to_p3(h, &r); + } + + ge25519_p3_dbl(&r, h); + ge25519_p1p1_to_p2(&s, &r); + ge25519_p2_dbl(&r, &s); + ge25519_p1p1_to_p2(&s, &r); + ge25519_p2_dbl(&r, &s); + ge25519_p1p1_to_p2(&s, &r); + ge25519_p2_dbl(&r, &s); + ge25519_p1p1_to_p3(h, &r); + + for (i = 0; i < 64; i += 2) { + ge25519_select_base(&t, i / 2, e[i]); + ge25519_madd(&r, h, &t); + ge25519_p1p1_to_p3(h, &r); + } +} + +/* multiply by the order of the main subgroup l = 2^252+27742317777372353535851937790883648493 */ +static void +ge25519_mul_l(ge25519_p3 *r, const ge25519_p3 *A) +{ + static const signed char aslide[253] = { + 13, 0, 0, 0, 0, -1, 0, 0, 0, 0, -11, 0, 0, 0, 0, 0, 0, -5, 0, 0, 0, 0, 0, 0, -3, 0, 0, 0, 0, -13, 0, 0, 0, 0, 7, 0, 0, 0, 0, 0, 3, 0, 0, 0, 0, -13, 0, 0, 0, 0, 5, 0, 0, 0, 0, 0, 0, 0, 0, 11, 0, 0, 0, 0, 0, 11, 0, 0, 0, 0, -13, 0, 0, 0, 0, 0, 0, -3, 0, 0, 0, 0, 0, -1, 0, 0, 0, 0, 3, 0, 0, 0, 0, -11, 0, 0, 0, 0, 0, 0, 0, 15, 0, 0, 0, 0, 0, -1, 0, 0, 0, 0, -1, 0, 0, 0, 0, 7, 0, 0, 0, 0, 5, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1 + }; + ge25519_cached Ai[8]; + ge25519_p1p1 t; + ge25519_p3 u; + ge25519_p3 A2; + int i; + + ge25519_p3_to_cached(&Ai[0], A); + ge25519_p3_dbl(&t, A); + ge25519_p1p1_to_p3(&A2, &t); + ge25519_add(&t, &A2, &Ai[0]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[1], &u); + ge25519_add(&t, &A2, &Ai[1]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[2], &u); + ge25519_add(&t, &A2, &Ai[2]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[3], &u); + ge25519_add(&t, &A2, &Ai[3]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[4], &u); + ge25519_add(&t, &A2, &Ai[4]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[5], &u); + ge25519_add(&t, &A2, &Ai[5]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[6], &u); + ge25519_add(&t, &A2, &Ai[6]); + ge25519_p1p1_to_p3(&u, &t); + ge25519_p3_to_cached(&Ai[7], &u); + + ge25519_p3_0(r); + + for (i = 252; i >= 0; --i) { + ge25519_p3_dbl(&t, r); + + if (aslide[i] > 0) { + ge25519_p1p1_to_p3(&u, &t); + ge25519_add(&t, &u, &Ai[aslide[i] / 2]); + } else if (aslide[i] < 0) { + ge25519_p1p1_to_p3(&u, &t); + ge25519_sub(&t, &u, &Ai[(-aslide[i]) / 2]); + } + + ge25519_p1p1_to_p3(r, &t); + } +} + +int +ge25519_is_on_curve(const ge25519_p3 *p) +{ + fe25519 x2; + fe25519 y2; + fe25519 z2; + fe25519 z4; + fe25519 t0; + fe25519 t1; + + fe25519_sq(x2, p->X); + fe25519_sq(y2, p->Y); + fe25519_sq(z2, p->Z); + fe25519_sub(t0, y2, x2); + fe25519_mul(t0, t0, z2); + + fe25519_mul(t1, x2, y2); + fe25519_mul(t1, t1, d); + fe25519_sq(z4, z2); + fe25519_add(t1, t1, z4); + fe25519_sub(t0, t0, t1); + + return fe25519_iszero(t0); +} + +int +ge25519_is_on_main_subgroup(const ge25519_p3 *p) +{ + ge25519_p3 pl; + + ge25519_mul_l(&pl, p); + + return fe25519_iszero(pl.X); +} + +int +ge25519_is_canonical(const unsigned char *s) +{ + unsigned char c; + unsigned char local_d; + unsigned int i; + + c = (s[31] & 0x7f) ^ 0x7f; + for (i = 30; i > 0; i--) { + c |= s[i] ^ 0xff; + } + c = (((unsigned int) c) - 1U) >> 8; + local_d = (0xed - 1U - (unsigned int) s[0]) >> 8; + + return 1 - (c & local_d & 1); +} + +int +ge25519_has_small_order(const unsigned char s[32]) +{ + CRYPTO_ALIGN(16) + static const unsigned char blacklist[][32] = { + /* 0 (order 4) */ + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, + /* 1 (order 1) */ + { 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, + /* 2707385501144840649318225287225658788936804267575313519463743609750303402022 + (order 8) */ + { 0x26, 0xe8, 0x95, 0x8f, 0xc2, 0xb2, 0x27, 0xb0, 0x45, 0xc3, 0xf4, + 0x89, 0xf2, 0xef, 0x98, 0xf0, 0xd5, 0xdf, 0xac, 0x05, 0xd3, 0xc6, + 0x33, 0x39, 0xb1, 0x38, 0x02, 0x88, 0x6d, 0x53, 0xfc, 0x05 }, + /* 55188659117513257062467267217118295137698188065244968500265048394206261417927 + (order 8) */ + { 0xc7, 0x17, 0x6a, 0x70, 0x3d, 0x4d, 0xd8, 0x4f, 0xba, 0x3c, 0x0b, + 0x76, 0x0d, 0x10, 0x67, 0x0f, 0x2a, 0x20, 0x53, 0xfa, 0x2c, 0x39, + 0xcc, 0xc6, 0x4e, 0xc7, 0xfd, 0x77, 0x92, 0xac, 0x03, 0x7a }, + /* p-1 (order 2) */ + { 0xec, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x7f }, + /* p (=0, order 4) */ + { 0xed, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x7f }, + /* p+1 (=1, order 1) */ + { 0xee, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x7f } + }; + unsigned char c[7] = { 0 }; + unsigned int k; + size_t i, j; + + COMPILER_ASSERT(7 == sizeof blacklist / sizeof blacklist[0]); + for (j = 0; j < 31; j++) { + for (i = 0; i < sizeof blacklist / sizeof blacklist[0]; i++) { + c[i] |= s[j] ^ blacklist[i][j]; + } + } + for (i = 0; i < sizeof blacklist / sizeof blacklist[0]; i++) { + c[i] |= (s[j] & 0x7f) ^ blacklist[i][j]; + } + k = 0; + for (i = 0; i < sizeof blacklist / sizeof blacklist[0]; i++) { + k |= (c[i] - 1); + } + return (int) ((k >> 8) & 1); +} + +/* + Input: + a[0]+256*a[1]+...+256^31*a[31] = a + b[0]+256*b[1]+...+256^31*b[31] = b + * + Output: + s[0]+256*s[1]+...+256^31*s[31] = (ab) mod l + where l = 2^252 + 27742317777372353535851937790883648493. + */ + +static void +sc25519_mul(unsigned char s[32], const unsigned char a[32], const unsigned char b[32]) +{ + int64_t a0 = 2097151 & load_3(a); + int64_t a1 = 2097151 & (load_4(a + 2) >> 5); + int64_t a2 = 2097151 & (load_3(a + 5) >> 2); + int64_t a3 = 2097151 & (load_4(a + 7) >> 7); + int64_t a4 = 2097151 & (load_4(a + 10) >> 4); + int64_t a5 = 2097151 & (load_3(a + 13) >> 1); + int64_t a6 = 2097151 & (load_4(a + 15) >> 6); + int64_t a7 = 2097151 & (load_3(a + 18) >> 3); + int64_t a8 = 2097151 & load_3(a + 21); + int64_t a9 = 2097151 & (load_4(a + 23) >> 5); + int64_t a10 = 2097151 & (load_3(a + 26) >> 2); + int64_t a11 = (load_4(a + 28) >> 7); + + int64_t b0 = 2097151 & load_3(b); + int64_t b1 = 2097151 & (load_4(b + 2) >> 5); + int64_t b2 = 2097151 & (load_3(b + 5) >> 2); + int64_t b3 = 2097151 & (load_4(b + 7) >> 7); + int64_t b4 = 2097151 & (load_4(b + 10) >> 4); + int64_t b5 = 2097151 & (load_3(b + 13) >> 1); + int64_t b6 = 2097151 & (load_4(b + 15) >> 6); + int64_t b7 = 2097151 & (load_3(b + 18) >> 3); + int64_t b8 = 2097151 & load_3(b + 21); + int64_t b9 = 2097151 & (load_4(b + 23) >> 5); + int64_t b10 = 2097151 & (load_3(b + 26) >> 2); + int64_t b11 = (load_4(b + 28) >> 7); + + int64_t s0; + int64_t s1; + int64_t s2; + int64_t s3; + int64_t s4; + int64_t s5; + int64_t s6; + int64_t s7; + int64_t s8; + int64_t s9; + int64_t s10; + int64_t s11; + int64_t s12; + int64_t s13; + int64_t s14; + int64_t s15; + int64_t s16; + int64_t s17; + int64_t s18; + int64_t s19; + int64_t s20; + int64_t s21; + int64_t s22; + int64_t s23; + + int64_t carry0; + int64_t carry1; + int64_t carry2; + int64_t carry3; + int64_t carry4; + int64_t carry5; + int64_t carry6; + int64_t carry7; + int64_t carry8; + int64_t carry9; + int64_t carry10; + int64_t carry11; + int64_t carry12; + int64_t carry13; + int64_t carry14; + int64_t carry15; + int64_t carry16; + int64_t carry17; + int64_t carry18; + int64_t carry19; + int64_t carry20; + int64_t carry21; + int64_t carry22; + + s0 = a0 * b0; + s1 = a0 * b1 + a1 * b0; + s2 = a0 * b2 + a1 * b1 + a2 * b0; + s3 = a0 * b3 + a1 * b2 + a2 * b1 + a3 * b0; + s4 = a0 * b4 + a1 * b3 + a2 * b2 + a3 * b1 + a4 * b0; + s5 = a0 * b5 + a1 * b4 + a2 * b3 + a3 * b2 + a4 * b1 + a5 * b0; + s6 = a0 * b6 + a1 * b5 + a2 * b4 + a3 * b3 + a4 * b2 + a5 * b1 + a6 * b0; + s7 = a0 * b7 + a1 * b6 + a2 * b5 + a3 * b4 + a4 * b3 + a5 * b2 + + a6 * b1 + a7 * b0; + s8 = a0 * b8 + a1 * b7 + a2 * b6 + a3 * b5 + a4 * b4 + a5 * b3 + + a6 * b2 + a7 * b1 + a8 * b0; + s9 = a0 * b9 + a1 * b8 + a2 * b7 + a3 * b6 + a4 * b5 + a5 * b4 + + a6 * b3 + a7 * b2 + a8 * b1 + a9 * b0; + s10 = a0 * b10 + a1 * b9 + a2 * b8 + a3 * b7 + a4 * b6 + a5 * b5 + + a6 * b4 + a7 * b3 + a8 * b2 + a9 * b1 + a10 * b0; + s11 = a0 * b11 + a1 * b10 + a2 * b9 + a3 * b8 + a4 * b7 + a5 * b6 + + a6 * b5 + a7 * b4 + a8 * b3 + a9 * b2 + a10 * b1 + a11 * b0; + s12 = a1 * b11 + a2 * b10 + a3 * b9 + a4 * b8 + a5 * b7 + a6 * b6 + + a7 * b5 + a8 * b4 + a9 * b3 + a10 * b2 + a11 * b1; + s13 = a2 * b11 + a3 * b10 + a4 * b9 + a5 * b8 + a6 * b7 + a7 * b6 + + a8 * b5 + a9 * b4 + a10 * b3 + a11 * b2; + s14 = a3 * b11 + a4 * b10 + a5 * b9 + a6 * b8 + a7 * b7 + a8 * b6 + + a9 * b5 + a10 * b4 + a11 * b3; + s15 = a4 * b11 + a5 * b10 + a6 * b9 + a7 * b8 + a8 * b7 + a9 * b6 + + a10 * b5 + a11 * b4; + s16 = + a5 * b11 + a6 * b10 + a7 * b9 + a8 * b8 + a9 * b7 + a10 * b6 + a11 * b5; + s17 = a6 * b11 + a7 * b10 + a8 * b9 + a9 * b8 + a10 * b7 + a11 * b6; + s18 = a7 * b11 + a8 * b10 + a9 * b9 + a10 * b8 + a11 * b7; + s19 = a8 * b11 + a9 * b10 + a10 * b9 + a11 * b8; + s20 = a9 * b11 + a10 * b10 + a11 * b9; + s21 = a10 * b11 + a11 * b10; + s22 = a11 * b11; + s23 = 0; + + carry0 = (s0 + (int64_t) (1L << 20)) >> 21; + s1 += carry0; + s0 -= carry0 * ((uint64_t) 1L << 21); + carry2 = (s2 + (int64_t) (1L << 20)) >> 21; + s3 += carry2; + s2 -= carry2 * ((uint64_t) 1L << 21); + carry4 = (s4 + (int64_t) (1L << 20)) >> 21; + s5 += carry4; + s4 -= carry4 * ((uint64_t) 1L << 21); + carry6 = (s6 + (int64_t) (1L << 20)) >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry8 = (s8 + (int64_t) (1L << 20)) >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry10 = (s10 + (int64_t) (1L << 20)) >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + carry12 = (s12 + (int64_t) (1L << 20)) >> 21; + s13 += carry12; + s12 -= carry12 * ((uint64_t) 1L << 21); + carry14 = (s14 + (int64_t) (1L << 20)) >> 21; + s15 += carry14; + s14 -= carry14 * ((uint64_t) 1L << 21); + carry16 = (s16 + (int64_t) (1L << 20)) >> 21; + s17 += carry16; + s16 -= carry16 * ((uint64_t) 1L << 21); + carry18 = (s18 + (int64_t) (1L << 20)) >> 21; + s19 += carry18; + s18 -= carry18 * ((uint64_t) 1L << 21); + carry20 = (s20 + (int64_t) (1L << 20)) >> 21; + s21 += carry20; + s20 -= carry20 * ((uint64_t) 1L << 21); + carry22 = (s22 + (int64_t) (1L << 20)) >> 21; + s23 += carry22; + s22 -= carry22 * ((uint64_t) 1L << 21); + + carry1 = (s1 + (int64_t) (1L << 20)) >> 21; + s2 += carry1; + s1 -= carry1 * ((uint64_t) 1L << 21); + carry3 = (s3 + (int64_t) (1L << 20)) >> 21; + s4 += carry3; + s3 -= carry3 * ((uint64_t) 1L << 21); + carry5 = (s5 + (int64_t) (1L << 20)) >> 21; + s6 += carry5; + s5 -= carry5 * ((uint64_t) 1L << 21); + carry7 = (s7 + (int64_t) (1L << 20)) >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry9 = (s9 + (int64_t) (1L << 20)) >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry11 = (s11 + (int64_t) (1L << 20)) >> 21; + s12 += carry11; + s11 -= carry11 * ((uint64_t) 1L << 21); + carry13 = (s13 + (int64_t) (1L << 20)) >> 21; + s14 += carry13; + s13 -= carry13 * ((uint64_t) 1L << 21); + carry15 = (s15 + (int64_t) (1L << 20)) >> 21; + s16 += carry15; + s15 -= carry15 * ((uint64_t) 1L << 21); + carry17 = (s17 + (int64_t) (1L << 20)) >> 21; + s18 += carry17; + s17 -= carry17 * ((uint64_t) 1L << 21); + carry19 = (s19 + (int64_t) (1L << 20)) >> 21; + s20 += carry19; + s19 -= carry19 * ((uint64_t) 1L << 21); + carry21 = (s21 + (int64_t) (1L << 20)) >> 21; + s22 += carry21; + s21 -= carry21 * ((uint64_t) 1L << 21); + + s11 += s23 * 666643; + s12 += s23 * 470296; + s13 += s23 * 654183; + s14 -= s23 * 997805; + s15 += s23 * 136657; + s16 -= s23 * 683901; + + s10 += s22 * 666643; + s11 += s22 * 470296; + s12 += s22 * 654183; + s13 -= s22 * 997805; + s14 += s22 * 136657; + s15 -= s22 * 683901; + + s9 += s21 * 666643; + s10 += s21 * 470296; + s11 += s21 * 654183; + s12 -= s21 * 997805; + s13 += s21 * 136657; + s14 -= s21 * 683901; + + s8 += s20 * 666643; + s9 += s20 * 470296; + s10 += s20 * 654183; + s11 -= s20 * 997805; + s12 += s20 * 136657; + s13 -= s20 * 683901; + + s7 += s19 * 666643; + s8 += s19 * 470296; + s9 += s19 * 654183; + s10 -= s19 * 997805; + s11 += s19 * 136657; + s12 -= s19 * 683901; + + s6 += s18 * 666643; + s7 += s18 * 470296; + s8 += s18 * 654183; + s9 -= s18 * 997805; + s10 += s18 * 136657; + s11 -= s18 * 683901; + + carry6 = (s6 + (int64_t) (1L << 20)) >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry8 = (s8 + (int64_t) (1L << 20)) >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry10 = (s10 + (int64_t) (1L << 20)) >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + carry12 = (s12 + (int64_t) (1L << 20)) >> 21; + s13 += carry12; + s12 -= carry12 * ((uint64_t) 1L << 21); + carry14 = (s14 + (int64_t) (1L << 20)) >> 21; + s15 += carry14; + s14 -= carry14 * ((uint64_t) 1L << 21); + carry16 = (s16 + (int64_t) (1L << 20)) >> 21; + s17 += carry16; + s16 -= carry16 * ((uint64_t) 1L << 21); + + carry7 = (s7 + (int64_t) (1L << 20)) >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry9 = (s9 + (int64_t) (1L << 20)) >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry11 = (s11 + (int64_t) (1L << 20)) >> 21; + s12 += carry11; + s11 -= carry11 * ((uint64_t) 1L << 21); + carry13 = (s13 + (int64_t) (1L << 20)) >> 21; + s14 += carry13; + s13 -= carry13 * ((uint64_t) 1L << 21); + carry15 = (s15 + (int64_t) (1L << 20)) >> 21; + s16 += carry15; + s15 -= carry15 * ((uint64_t) 1L << 21); + + s5 += s17 * 666643; + s6 += s17 * 470296; + s7 += s17 * 654183; + s8 -= s17 * 997805; + s9 += s17 * 136657; + s10 -= s17 * 683901; + + s4 += s16 * 666643; + s5 += s16 * 470296; + s6 += s16 * 654183; + s7 -= s16 * 997805; + s8 += s16 * 136657; + s9 -= s16 * 683901; + + s3 += s15 * 666643; + s4 += s15 * 470296; + s5 += s15 * 654183; + s6 -= s15 * 997805; + s7 += s15 * 136657; + s8 -= s15 * 683901; + + s2 += s14 * 666643; + s3 += s14 * 470296; + s4 += s14 * 654183; + s5 -= s14 * 997805; + s6 += s14 * 136657; + s7 -= s14 * 683901; + + s1 += s13 * 666643; + s2 += s13 * 470296; + s3 += s13 * 654183; + s4 -= s13 * 997805; + s5 += s13 * 136657; + s6 -= s13 * 683901; + + s0 += s12 * 666643; + s1 += s12 * 470296; + s2 += s12 * 654183; + s3 -= s12 * 997805; + s4 += s12 * 136657; + s5 -= s12 * 683901; + s12 = 0; + + carry0 = (s0 + (int64_t) (1L << 20)) >> 21; + s1 += carry0; + s0 -= carry0 * ((uint64_t) 1L << 21); + carry2 = (s2 + (int64_t) (1L << 20)) >> 21; + s3 += carry2; + s2 -= carry2 * ((uint64_t) 1L << 21); + carry4 = (s4 + (int64_t) (1L << 20)) >> 21; + s5 += carry4; + s4 -= carry4 * ((uint64_t) 1L << 21); + carry6 = (s6 + (int64_t) (1L << 20)) >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry8 = (s8 + (int64_t) (1L << 20)) >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry10 = (s10 + (int64_t) (1L << 20)) >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + + carry1 = (s1 + (int64_t) (1L << 20)) >> 21; + s2 += carry1; + s1 -= carry1 * ((uint64_t) 1L << 21); + carry3 = (s3 + (int64_t) (1L << 20)) >> 21; + s4 += carry3; + s3 -= carry3 * ((uint64_t) 1L << 21); + carry5 = (s5 + (int64_t) (1L << 20)) >> 21; + s6 += carry5; + s5 -= carry5 * ((uint64_t) 1L << 21); + carry7 = (s7 + (int64_t) (1L << 20)) >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry9 = (s9 + (int64_t) (1L << 20)) >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry11 = (s11 + (int64_t) (1L << 20)) >> 21; + s12 += carry11; + s11 -= carry11 * ((uint64_t) 1L << 21); + + s0 += s12 * 666643; + s1 += s12 * 470296; + s2 += s12 * 654183; + s3 -= s12 * 997805; + s4 += s12 * 136657; + s5 -= s12 * 683901; + s12 = 0; + + carry0 = s0 >> 21; + s1 += carry0; + s0 -= carry0 * ((uint64_t) 1L << 21); + carry1 = s1 >> 21; + s2 += carry1; + s1 -= carry1 * ((uint64_t) 1L << 21); + carry2 = s2 >> 21; + s3 += carry2; + s2 -= carry2 * ((uint64_t) 1L << 21); + carry3 = s3 >> 21; + s4 += carry3; + s3 -= carry3 * ((uint64_t) 1L << 21); + carry4 = s4 >> 21; + s5 += carry4; + s4 -= carry4 * ((uint64_t) 1L << 21); + carry5 = s5 >> 21; + s6 += carry5; + s5 -= carry5 * ((uint64_t) 1L << 21); + carry6 = s6 >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry7 = s7 >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry8 = s8 >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry9 = s9 >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry10 = s10 >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + carry11 = s11 >> 21; + s12 += carry11; + s11 -= carry11 * ((uint64_t) 1L << 21); + + s0 += s12 * 666643; + s1 += s12 * 470296; + s2 += s12 * 654183; + s3 -= s12 * 997805; + s4 += s12 * 136657; + s5 -= s12 * 683901; + + carry0 = s0 >> 21; + s1 += carry0; + s0 -= carry0 * ((uint64_t) 1L << 21); + carry1 = s1 >> 21; + s2 += carry1; + s1 -= carry1 * ((uint64_t) 1L << 21); + carry2 = s2 >> 21; + s3 += carry2; + s2 -= carry2 * ((uint64_t) 1L << 21); + carry3 = s3 >> 21; + s4 += carry3; + s3 -= carry3 * ((uint64_t) 1L << 21); + carry4 = s4 >> 21; + s5 += carry4; + s4 -= carry4 * ((uint64_t) 1L << 21); + carry5 = s5 >> 21; + s6 += carry5; + s5 -= carry5 * ((uint64_t) 1L << 21); + carry6 = s6 >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry7 = s7 >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry8 = s8 >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry9 = s9 >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry10 = s10 >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + + s[0] = s0 >> 0; + s[1] = s0 >> 8; + s[2] = (s0 >> 16) | (s1 * ((uint64_t) 1 << 5)); + s[3] = s1 >> 3; + s[4] = s1 >> 11; + s[5] = (s1 >> 19) | (s2 * ((uint64_t) 1 << 2)); + s[6] = s2 >> 6; + s[7] = (s2 >> 14) | (s3 * ((uint64_t) 1 << 7)); + s[8] = s3 >> 1; + s[9] = s3 >> 9; + s[10] = (s3 >> 17) | (s4 * ((uint64_t) 1 << 4)); + s[11] = s4 >> 4; + s[12] = s4 >> 12; + s[13] = (s4 >> 20) | (s5 * ((uint64_t) 1 << 1)); + s[14] = s5 >> 7; + s[15] = (s5 >> 15) | (s6 * ((uint64_t) 1 << 6)); + s[16] = s6 >> 2; + s[17] = s6 >> 10; + s[18] = (s6 >> 18) | (s7 * ((uint64_t) 1 << 3)); + s[19] = s7 >> 5; + s[20] = s7 >> 13; + s[21] = s8 >> 0; + s[22] = s8 >> 8; + s[23] = (s8 >> 16) | (s9 * ((uint64_t) 1 << 5)); + s[24] = s9 >> 3; + s[25] = s9 >> 11; + s[26] = (s9 >> 19) | (s10 * ((uint64_t) 1 << 2)); + s[27] = s10 >> 6; + s[28] = (s10 >> 14) | (s11 * ((uint64_t) 1 << 7)); + s[29] = s11 >> 1; + s[30] = s11 >> 9; + s[31] = s11 >> 17; +} + +/* + Input: + a[0]+256*a[1]+...+256^31*a[31] = a + b[0]+256*b[1]+...+256^31*b[31] = b + c[0]+256*c[1]+...+256^31*c[31] = c + * + Output: + s[0]+256*s[1]+...+256^31*s[31] = (ab+c) mod l + where l = 2^252 + 27742317777372353535851937790883648493. + */ + +void +sc25519_muladd(unsigned char s[32], const unsigned char a[32], + const unsigned char b[32], const unsigned char c[32]) +{ + int64_t a0 = 2097151 & load_3(a); + int64_t a1 = 2097151 & (load_4(a + 2) >> 5); + int64_t a2 = 2097151 & (load_3(a + 5) >> 2); + int64_t a3 = 2097151 & (load_4(a + 7) >> 7); + int64_t a4 = 2097151 & (load_4(a + 10) >> 4); + int64_t a5 = 2097151 & (load_3(a + 13) >> 1); + int64_t a6 = 2097151 & (load_4(a + 15) >> 6); + int64_t a7 = 2097151 & (load_3(a + 18) >> 3); + int64_t a8 = 2097151 & load_3(a + 21); + int64_t a9 = 2097151 & (load_4(a + 23) >> 5); + int64_t a10 = 2097151 & (load_3(a + 26) >> 2); + int64_t a11 = (load_4(a + 28) >> 7); + + int64_t b0 = 2097151 & load_3(b); + int64_t b1 = 2097151 & (load_4(b + 2) >> 5); + int64_t b2 = 2097151 & (load_3(b + 5) >> 2); + int64_t b3 = 2097151 & (load_4(b + 7) >> 7); + int64_t b4 = 2097151 & (load_4(b + 10) >> 4); + int64_t b5 = 2097151 & (load_3(b + 13) >> 1); + int64_t b6 = 2097151 & (load_4(b + 15) >> 6); + int64_t b7 = 2097151 & (load_3(b + 18) >> 3); + int64_t b8 = 2097151 & load_3(b + 21); + int64_t b9 = 2097151 & (load_4(b + 23) >> 5); + int64_t b10 = 2097151 & (load_3(b + 26) >> 2); + int64_t b11 = (load_4(b + 28) >> 7); + + int64_t c0 = 2097151 & load_3(c); + int64_t c1 = 2097151 & (load_4(c + 2) >> 5); + int64_t c2 = 2097151 & (load_3(c + 5) >> 2); + int64_t c3 = 2097151 & (load_4(c + 7) >> 7); + int64_t c4 = 2097151 & (load_4(c + 10) >> 4); + int64_t c5 = 2097151 & (load_3(c + 13) >> 1); + int64_t c6 = 2097151 & (load_4(c + 15) >> 6); + int64_t c7 = 2097151 & (load_3(c + 18) >> 3); + int64_t c8 = 2097151 & load_3(c + 21); + int64_t c9 = 2097151 & (load_4(c + 23) >> 5); + int64_t c10 = 2097151 & (load_3(c + 26) >> 2); + int64_t c11 = (load_4(c + 28) >> 7); + + int64_t s0; + int64_t s1; + int64_t s2; + int64_t s3; + int64_t s4; + int64_t s5; + int64_t s6; + int64_t s7; + int64_t s8; + int64_t s9; + int64_t s10; + int64_t s11; + int64_t s12; + int64_t s13; + int64_t s14; + int64_t s15; + int64_t s16; + int64_t s17; + int64_t s18; + int64_t s19; + int64_t s20; + int64_t s21; + int64_t s22; + int64_t s23; + + int64_t carry0; + int64_t carry1; + int64_t carry2; + int64_t carry3; + int64_t carry4; + int64_t carry5; + int64_t carry6; + int64_t carry7; + int64_t carry8; + int64_t carry9; + int64_t carry10; + int64_t carry11; + int64_t carry12; + int64_t carry13; + int64_t carry14; + int64_t carry15; + int64_t carry16; + int64_t carry17; + int64_t carry18; + int64_t carry19; + int64_t carry20; + int64_t carry21; + int64_t carry22; + + s0 = c0 + a0 * b0; + s1 = c1 + a0 * b1 + a1 * b0; + s2 = c2 + a0 * b2 + a1 * b1 + a2 * b0; + s3 = c3 + a0 * b3 + a1 * b2 + a2 * b1 + a3 * b0; + s4 = c4 + a0 * b4 + a1 * b3 + a2 * b2 + a3 * b1 + a4 * b0; + s5 = c5 + a0 * b5 + a1 * b4 + a2 * b3 + a3 * b2 + a4 * b1 + a5 * b0; + s6 = c6 + a0 * b6 + a1 * b5 + a2 * b4 + a3 * b3 + a4 * b2 + a5 * b1 + + a6 * b0; + s7 = c7 + a0 * b7 + a1 * b6 + a2 * b5 + a3 * b4 + a4 * b3 + a5 * b2 + + a6 * b1 + a7 * b0; + s8 = c8 + a0 * b8 + a1 * b7 + a2 * b6 + a3 * b5 + a4 * b4 + a5 * b3 + + a6 * b2 + a7 * b1 + a8 * b0; + s9 = c9 + a0 * b9 + a1 * b8 + a2 * b7 + a3 * b6 + a4 * b5 + a5 * b4 + + a6 * b3 + a7 * b2 + a8 * b1 + a9 * b0; + s10 = c10 + a0 * b10 + a1 * b9 + a2 * b8 + a3 * b7 + a4 * b6 + a5 * b5 + + a6 * b4 + a7 * b3 + a8 * b2 + a9 * b1 + a10 * b0; + s11 = c11 + a0 * b11 + a1 * b10 + a2 * b9 + a3 * b8 + a4 * b7 + a5 * b6 + + a6 * b5 + a7 * b4 + a8 * b3 + a9 * b2 + a10 * b1 + a11 * b0; + s12 = a1 * b11 + a2 * b10 + a3 * b9 + a4 * b8 + a5 * b7 + a6 * b6 + + a7 * b5 + a8 * b4 + a9 * b3 + a10 * b2 + a11 * b1; + s13 = a2 * b11 + a3 * b10 + a4 * b9 + a5 * b8 + a6 * b7 + a7 * b6 + + a8 * b5 + a9 * b4 + a10 * b3 + a11 * b2; + s14 = a3 * b11 + a4 * b10 + a5 * b9 + a6 * b8 + a7 * b7 + a8 * b6 + + a9 * b5 + a10 * b4 + a11 * b3; + s15 = a4 * b11 + a5 * b10 + a6 * b9 + a7 * b8 + a8 * b7 + a9 * b6 + + a10 * b5 + a11 * b4; + s16 = + a5 * b11 + a6 * b10 + a7 * b9 + a8 * b8 + a9 * b7 + a10 * b6 + a11 * b5; + s17 = a6 * b11 + a7 * b10 + a8 * b9 + a9 * b8 + a10 * b7 + a11 * b6; + s18 = a7 * b11 + a8 * b10 + a9 * b9 + a10 * b8 + a11 * b7; + s19 = a8 * b11 + a9 * b10 + a10 * b9 + a11 * b8; + s20 = a9 * b11 + a10 * b10 + a11 * b9; + s21 = a10 * b11 + a11 * b10; + s22 = a11 * b11; + s23 = 0; + + carry0 = (s0 + (int64_t) (1L << 20)) >> 21; + s1 += carry0; + s0 -= carry0 * ((uint64_t) 1L << 21); + carry2 = (s2 + (int64_t) (1L << 20)) >> 21; + s3 += carry2; + s2 -= carry2 * ((uint64_t) 1L << 21); + carry4 = (s4 + (int64_t) (1L << 20)) >> 21; + s5 += carry4; + s4 -= carry4 * ((uint64_t) 1L << 21); + carry6 = (s6 + (int64_t) (1L << 20)) >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry8 = (s8 + (int64_t) (1L << 20)) >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry10 = (s10 + (int64_t) (1L << 20)) >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + carry12 = (s12 + (int64_t) (1L << 20)) >> 21; + s13 += carry12; + s12 -= carry12 * ((uint64_t) 1L << 21); + carry14 = (s14 + (int64_t) (1L << 20)) >> 21; + s15 += carry14; + s14 -= carry14 * ((uint64_t) 1L << 21); + carry16 = (s16 + (int64_t) (1L << 20)) >> 21; + s17 += carry16; + s16 -= carry16 * ((uint64_t) 1L << 21); + carry18 = (s18 + (int64_t) (1L << 20)) >> 21; + s19 += carry18; + s18 -= carry18 * ((uint64_t) 1L << 21); + carry20 = (s20 + (int64_t) (1L << 20)) >> 21; + s21 += carry20; + s20 -= carry20 * ((uint64_t) 1L << 21); + carry22 = (s22 + (int64_t) (1L << 20)) >> 21; + s23 += carry22; + s22 -= carry22 * ((uint64_t) 1L << 21); + + carry1 = (s1 + (int64_t) (1L << 20)) >> 21; + s2 += carry1; + s1 -= carry1 * ((uint64_t) 1L << 21); + carry3 = (s3 + (int64_t) (1L << 20)) >> 21; + s4 += carry3; + s3 -= carry3 * ((uint64_t) 1L << 21); + carry5 = (s5 + (int64_t) (1L << 20)) >> 21; + s6 += carry5; + s5 -= carry5 * ((uint64_t) 1L << 21); + carry7 = (s7 + (int64_t) (1L << 20)) >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry9 = (s9 + (int64_t) (1L << 20)) >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry11 = (s11 + (int64_t) (1L << 20)) >> 21; + s12 += carry11; + s11 -= carry11 * ((uint64_t) 1L << 21); + carry13 = (s13 + (int64_t) (1L << 20)) >> 21; + s14 += carry13; + s13 -= carry13 * ((uint64_t) 1L << 21); + carry15 = (s15 + (int64_t) (1L << 20)) >> 21; + s16 += carry15; + s15 -= carry15 * ((uint64_t) 1L << 21); + carry17 = (s17 + (int64_t) (1L << 20)) >> 21; + s18 += carry17; + s17 -= carry17 * ((uint64_t) 1L << 21); + carry19 = (s19 + (int64_t) (1L << 20)) >> 21; + s20 += carry19; + s19 -= carry19 * ((uint64_t) 1L << 21); + carry21 = (s21 + (int64_t) (1L << 20)) >> 21; + s22 += carry21; + s21 -= carry21 * ((uint64_t) 1L << 21); + + s11 += s23 * 666643; + s12 += s23 * 470296; + s13 += s23 * 654183; + s14 -= s23 * 997805; + s15 += s23 * 136657; + s16 -= s23 * 683901; + + s10 += s22 * 666643; + s11 += s22 * 470296; + s12 += s22 * 654183; + s13 -= s22 * 997805; + s14 += s22 * 136657; + s15 -= s22 * 683901; + + s9 += s21 * 666643; + s10 += s21 * 470296; + s11 += s21 * 654183; + s12 -= s21 * 997805; + s13 += s21 * 136657; + s14 -= s21 * 683901; + + s8 += s20 * 666643; + s9 += s20 * 470296; + s10 += s20 * 654183; + s11 -= s20 * 997805; + s12 += s20 * 136657; + s13 -= s20 * 683901; + + s7 += s19 * 666643; + s8 += s19 * 470296; + s9 += s19 * 654183; + s10 -= s19 * 997805; + s11 += s19 * 136657; + s12 -= s19 * 683901; + + s6 += s18 * 666643; + s7 += s18 * 470296; + s8 += s18 * 654183; + s9 -= s18 * 997805; + s10 += s18 * 136657; + s11 -= s18 * 683901; + + carry6 = (s6 + (int64_t) (1L << 20)) >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry8 = (s8 + (int64_t) (1L << 20)) >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry10 = (s10 + (int64_t) (1L << 20)) >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + carry12 = (s12 + (int64_t) (1L << 20)) >> 21; + s13 += carry12; + s12 -= carry12 * ((uint64_t) 1L << 21); + carry14 = (s14 + (int64_t) (1L << 20)) >> 21; + s15 += carry14; + s14 -= carry14 * ((uint64_t) 1L << 21); + carry16 = (s16 + (int64_t) (1L << 20)) >> 21; + s17 += carry16; + s16 -= carry16 * ((uint64_t) 1L << 21); + + carry7 = (s7 + (int64_t) (1L << 20)) >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry9 = (s9 + (int64_t) (1L << 20)) >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry11 = (s11 + (int64_t) (1L << 20)) >> 21; + s12 += carry11; + s11 -= carry11 * ((uint64_t) 1L << 21); + carry13 = (s13 + (int64_t) (1L << 20)) >> 21; + s14 += carry13; + s13 -= carry13 * ((uint64_t) 1L << 21); + carry15 = (s15 + (int64_t) (1L << 20)) >> 21; + s16 += carry15; + s15 -= carry15 * ((uint64_t) 1L << 21); + + s5 += s17 * 666643; + s6 += s17 * 470296; + s7 += s17 * 654183; + s8 -= s17 * 997805; + s9 += s17 * 136657; + s10 -= s17 * 683901; + + s4 += s16 * 666643; + s5 += s16 * 470296; + s6 += s16 * 654183; + s7 -= s16 * 997805; + s8 += s16 * 136657; + s9 -= s16 * 683901; + + s3 += s15 * 666643; + s4 += s15 * 470296; + s5 += s15 * 654183; + s6 -= s15 * 997805; + s7 += s15 * 136657; + s8 -= s15 * 683901; + + s2 += s14 * 666643; + s3 += s14 * 470296; + s4 += s14 * 654183; + s5 -= s14 * 997805; + s6 += s14 * 136657; + s7 -= s14 * 683901; + + s1 += s13 * 666643; + s2 += s13 * 470296; + s3 += s13 * 654183; + s4 -= s13 * 997805; + s5 += s13 * 136657; + s6 -= s13 * 683901; + + s0 += s12 * 666643; + s1 += s12 * 470296; + s2 += s12 * 654183; + s3 -= s12 * 997805; + s4 += s12 * 136657; + s5 -= s12 * 683901; + s12 = 0; + + carry0 = (s0 + (int64_t) (1L << 20)) >> 21; + s1 += carry0; + s0 -= carry0 * ((uint64_t) 1L << 21); + carry2 = (s2 + (int64_t) (1L << 20)) >> 21; + s3 += carry2; + s2 -= carry2 * ((uint64_t) 1L << 21); + carry4 = (s4 + (int64_t) (1L << 20)) >> 21; + s5 += carry4; + s4 -= carry4 * ((uint64_t) 1L << 21); + carry6 = (s6 + (int64_t) (1L << 20)) >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry8 = (s8 + (int64_t) (1L << 20)) >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry10 = (s10 + (int64_t) (1L << 20)) >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + + carry1 = (s1 + (int64_t) (1L << 20)) >> 21; + s2 += carry1; + s1 -= carry1 * ((uint64_t) 1L << 21); + carry3 = (s3 + (int64_t) (1L << 20)) >> 21; + s4 += carry3; + s3 -= carry3 * ((uint64_t) 1L << 21); + carry5 = (s5 + (int64_t) (1L << 20)) >> 21; + s6 += carry5; + s5 -= carry5 * ((uint64_t) 1L << 21); + carry7 = (s7 + (int64_t) (1L << 20)) >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry9 = (s9 + (int64_t) (1L << 20)) >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry11 = (s11 + (int64_t) (1L << 20)) >> 21; + s12 += carry11; + s11 -= carry11 * ((uint64_t) 1L << 21); + + s0 += s12 * 666643; + s1 += s12 * 470296; + s2 += s12 * 654183; + s3 -= s12 * 997805; + s4 += s12 * 136657; + s5 -= s12 * 683901; + s12 = 0; + + carry0 = s0 >> 21; + s1 += carry0; + s0 -= carry0 * ((uint64_t) 1L << 21); + carry1 = s1 >> 21; + s2 += carry1; + s1 -= carry1 * ((uint64_t) 1L << 21); + carry2 = s2 >> 21; + s3 += carry2; + s2 -= carry2 * ((uint64_t) 1L << 21); + carry3 = s3 >> 21; + s4 += carry3; + s3 -= carry3 * ((uint64_t) 1L << 21); + carry4 = s4 >> 21; + s5 += carry4; + s4 -= carry4 * ((uint64_t) 1L << 21); + carry5 = s5 >> 21; + s6 += carry5; + s5 -= carry5 * ((uint64_t) 1L << 21); + carry6 = s6 >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry7 = s7 >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry8 = s8 >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry9 = s9 >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry10 = s10 >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + carry11 = s11 >> 21; + s12 += carry11; + s11 -= carry11 * ((uint64_t) 1L << 21); + + s0 += s12 * 666643; + s1 += s12 * 470296; + s2 += s12 * 654183; + s3 -= s12 * 997805; + s4 += s12 * 136657; + s5 -= s12 * 683901; + + carry0 = s0 >> 21; + s1 += carry0; + s0 -= carry0 * ((uint64_t) 1L << 21); + carry1 = s1 >> 21; + s2 += carry1; + s1 -= carry1 * ((uint64_t) 1L << 21); + carry2 = s2 >> 21; + s3 += carry2; + s2 -= carry2 * ((uint64_t) 1L << 21); + carry3 = s3 >> 21; + s4 += carry3; + s3 -= carry3 * ((uint64_t) 1L << 21); + carry4 = s4 >> 21; + s5 += carry4; + s4 -= carry4 * ((uint64_t) 1L << 21); + carry5 = s5 >> 21; + s6 += carry5; + s5 -= carry5 * ((uint64_t) 1L << 21); + carry6 = s6 >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry7 = s7 >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry8 = s8 >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry9 = s9 >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry10 = s10 >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + + s[0] = s0 >> 0; + s[1] = s0 >> 8; + s[2] = (s0 >> 16) | (s1 * ((uint64_t) 1 << 5)); + s[3] = s1 >> 3; + s[4] = s1 >> 11; + s[5] = (s1 >> 19) | (s2 * ((uint64_t) 1 << 2)); + s[6] = s2 >> 6; + s[7] = (s2 >> 14) | (s3 * ((uint64_t) 1 << 7)); + s[8] = s3 >> 1; + s[9] = s3 >> 9; + s[10] = (s3 >> 17) | (s4 * ((uint64_t) 1 << 4)); + s[11] = s4 >> 4; + s[12] = s4 >> 12; + s[13] = (s4 >> 20) | (s5 * ((uint64_t) 1 << 1)); + s[14] = s5 >> 7; + s[15] = (s5 >> 15) | (s6 * ((uint64_t) 1 << 6)); + s[16] = s6 >> 2; + s[17] = s6 >> 10; + s[18] = (s6 >> 18) | (s7 * ((uint64_t) 1 << 3)); + s[19] = s7 >> 5; + s[20] = s7 >> 13; + s[21] = s8 >> 0; + s[22] = s8 >> 8; + s[23] = (s8 >> 16) | (s9 * ((uint64_t) 1 << 5)); + s[24] = s9 >> 3; + s[25] = s9 >> 11; + s[26] = (s9 >> 19) | (s10 * ((uint64_t) 1 << 2)); + s[27] = s10 >> 6; + s[28] = (s10 >> 14) | (s11 * ((uint64_t) 1 << 7)); + s[29] = s11 >> 1; + s[30] = s11 >> 9; + s[31] = s11 >> 17; +} + +/* + Input: + a[0]+256*a[1]+...+256^31*a[31] = a + * + Output: + s[0]+256*s[1]+...+256^31*s[31] = a^2 mod l + where l = 2^252 + 27742317777372353535851937790883648493. + */ + +static inline void +sc25519_sq(unsigned char *s, const unsigned char *a) +{ + sc25519_mul(s, a, a); +} + +/* + Input: + s[0]+256*a[1]+...+256^31*a[31] = a + n + * + Output: + s[0]+256*s[1]+...+256^31*s[31] = x * s^(s^n) mod l + where l = 2^252 + 27742317777372353535851937790883648493. + Overwrites s in place. + */ + +static inline void +sc25519_sqmul(unsigned char s[32], const int n, const unsigned char a[32]) +{ + int i; + + for (i = 0; i < n; i++) { + sc25519_sq(s, s); + } + sc25519_mul(s, s, a); +} + +void +sc25519_invert(unsigned char recip[32], const unsigned char s[32]) +{ + unsigned char _10[32], _100[32], _11[32], _101[32], _111[32], + _1001[32], _1011[32], _1111[32]; + + sc25519_sq(_10, s); + sc25519_sq(_100, _10); + sc25519_mul(_11, _10, s); + sc25519_mul(_101, _10, _11); + sc25519_mul(_111, _10, _101); + sc25519_mul(_1001, _10, _111); + sc25519_mul(_1011, _10, _1001); + sc25519_mul(_1111, _100, _1011); + sc25519_mul(recip, _1111, s); + + sc25519_sqmul(recip, 123 + 3, _101); + sc25519_sqmul(recip, 2 + 2, _11); + sc25519_sqmul(recip, 1 + 4, _1111); + sc25519_sqmul(recip, 1 + 4, _1111); + sc25519_sqmul(recip, 4, _1001); + sc25519_sqmul(recip, 2, _11); + sc25519_sqmul(recip, 1 + 4, _1111); + sc25519_sqmul(recip, 1 + 3, _101); + sc25519_sqmul(recip, 3 + 3, _101); + sc25519_sqmul(recip, 3, _111); + sc25519_sqmul(recip, 1 + 4, _1111); + sc25519_sqmul(recip, 2 + 3, _111); + sc25519_sqmul(recip, 2 + 2, _11); + sc25519_sqmul(recip, 1 + 4, _1011); + sc25519_sqmul(recip, 2 + 4, _1011); + sc25519_sqmul(recip, 6 + 4, _1001); + sc25519_sqmul(recip, 2 + 2, _11); + sc25519_sqmul(recip, 3 + 2, _11); + sc25519_sqmul(recip, 3 + 2, _11); + sc25519_sqmul(recip, 1 + 4, _1001); + sc25519_sqmul(recip, 1 + 3, _111); + sc25519_sqmul(recip, 2 + 4, _1111); + sc25519_sqmul(recip, 1 + 4, _1011); + sc25519_sqmul(recip, 3, _101); + sc25519_sqmul(recip, 2 + 4, _1111); + sc25519_sqmul(recip, 3, _101); + sc25519_sqmul(recip, 1 + 2, _11); +} + +/* + Input: + s[0]+256*s[1]+...+256^63*s[63] = s + * + Output: + s[0]+256*s[1]+...+256^31*s[31] = s mod l + where l = 2^252 + 27742317777372353535851937790883648493. + Overwrites s in place. + */ + +void +sc25519_reduce(unsigned char s[64]) +{ + int64_t s0 = 2097151 & load_3(s); + int64_t s1 = 2097151 & (load_4(s + 2) >> 5); + int64_t s2 = 2097151 & (load_3(s + 5) >> 2); + int64_t s3 = 2097151 & (load_4(s + 7) >> 7); + int64_t s4 = 2097151 & (load_4(s + 10) >> 4); + int64_t s5 = 2097151 & (load_3(s + 13) >> 1); + int64_t s6 = 2097151 & (load_4(s + 15) >> 6); + int64_t s7 = 2097151 & (load_3(s + 18) >> 3); + int64_t s8 = 2097151 & load_3(s + 21); + int64_t s9 = 2097151 & (load_4(s + 23) >> 5); + int64_t s10 = 2097151 & (load_3(s + 26) >> 2); + int64_t s11 = 2097151 & (load_4(s + 28) >> 7); + int64_t s12 = 2097151 & (load_4(s + 31) >> 4); + int64_t s13 = 2097151 & (load_3(s + 34) >> 1); + int64_t s14 = 2097151 & (load_4(s + 36) >> 6); + int64_t s15 = 2097151 & (load_3(s + 39) >> 3); + int64_t s16 = 2097151 & load_3(s + 42); + int64_t s17 = 2097151 & (load_4(s + 44) >> 5); + int64_t s18 = 2097151 & (load_3(s + 47) >> 2); + int64_t s19 = 2097151 & (load_4(s + 49) >> 7); + int64_t s20 = 2097151 & (load_4(s + 52) >> 4); + int64_t s21 = 2097151 & (load_3(s + 55) >> 1); + int64_t s22 = 2097151 & (load_4(s + 57) >> 6); + int64_t s23 = (load_4(s + 60) >> 3); + + int64_t carry0; + int64_t carry1; + int64_t carry2; + int64_t carry3; + int64_t carry4; + int64_t carry5; + int64_t carry6; + int64_t carry7; + int64_t carry8; + int64_t carry9; + int64_t carry10; + int64_t carry11; + int64_t carry12; + int64_t carry13; + int64_t carry14; + int64_t carry15; + int64_t carry16; + + s11 += s23 * 666643; + s12 += s23 * 470296; + s13 += s23 * 654183; + s14 -= s23 * 997805; + s15 += s23 * 136657; + s16 -= s23 * 683901; + + s10 += s22 * 666643; + s11 += s22 * 470296; + s12 += s22 * 654183; + s13 -= s22 * 997805; + s14 += s22 * 136657; + s15 -= s22 * 683901; + + s9 += s21 * 666643; + s10 += s21 * 470296; + s11 += s21 * 654183; + s12 -= s21 * 997805; + s13 += s21 * 136657; + s14 -= s21 * 683901; + + s8 += s20 * 666643; + s9 += s20 * 470296; + s10 += s20 * 654183; + s11 -= s20 * 997805; + s12 += s20 * 136657; + s13 -= s20 * 683901; + + s7 += s19 * 666643; + s8 += s19 * 470296; + s9 += s19 * 654183; + s10 -= s19 * 997805; + s11 += s19 * 136657; + s12 -= s19 * 683901; + + s6 += s18 * 666643; + s7 += s18 * 470296; + s8 += s18 * 654183; + s9 -= s18 * 997805; + s10 += s18 * 136657; + s11 -= s18 * 683901; + + carry6 = (s6 + (int64_t) (1L << 20)) >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry8 = (s8 + (int64_t) (1L << 20)) >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry10 = (s10 + (int64_t) (1L << 20)) >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + carry12 = (s12 + (int64_t) (1L << 20)) >> 21; + s13 += carry12; + s12 -= carry12 * ((uint64_t) 1L << 21); + carry14 = (s14 + (int64_t) (1L << 20)) >> 21; + s15 += carry14; + s14 -= carry14 * ((uint64_t) 1L << 21); + carry16 = (s16 + (int64_t) (1L << 20)) >> 21; + s17 += carry16; + s16 -= carry16 * ((uint64_t) 1L << 21); + + carry7 = (s7 + (int64_t) (1L << 20)) >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry9 = (s9 + (int64_t) (1L << 20)) >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry11 = (s11 + (int64_t) (1L << 20)) >> 21; + s12 += carry11; + s11 -= carry11 * ((uint64_t) 1L << 21); + carry13 = (s13 + (int64_t) (1L << 20)) >> 21; + s14 += carry13; + s13 -= carry13 * ((uint64_t) 1L << 21); + carry15 = (s15 + (int64_t) (1L << 20)) >> 21; + s16 += carry15; + s15 -= carry15 * ((uint64_t) 1L << 21); + + s5 += s17 * 666643; + s6 += s17 * 470296; + s7 += s17 * 654183; + s8 -= s17 * 997805; + s9 += s17 * 136657; + s10 -= s17 * 683901; + + s4 += s16 * 666643; + s5 += s16 * 470296; + s6 += s16 * 654183; + s7 -= s16 * 997805; + s8 += s16 * 136657; + s9 -= s16 * 683901; + + s3 += s15 * 666643; + s4 += s15 * 470296; + s5 += s15 * 654183; + s6 -= s15 * 997805; + s7 += s15 * 136657; + s8 -= s15 * 683901; + + s2 += s14 * 666643; + s3 += s14 * 470296; + s4 += s14 * 654183; + s5 -= s14 * 997805; + s6 += s14 * 136657; + s7 -= s14 * 683901; + + s1 += s13 * 666643; + s2 += s13 * 470296; + s3 += s13 * 654183; + s4 -= s13 * 997805; + s5 += s13 * 136657; + s6 -= s13 * 683901; + + s0 += s12 * 666643; + s1 += s12 * 470296; + s2 += s12 * 654183; + s3 -= s12 * 997805; + s4 += s12 * 136657; + s5 -= s12 * 683901; + s12 = 0; + + carry0 = (s0 + (int64_t) (1L << 20)) >> 21; + s1 += carry0; + s0 -= carry0 * ((uint64_t) 1L << 21); + carry2 = (s2 + (int64_t) (1L << 20)) >> 21; + s3 += carry2; + s2 -= carry2 * ((uint64_t) 1L << 21); + carry4 = (s4 + (int64_t) (1L << 20)) >> 21; + s5 += carry4; + s4 -= carry4 * ((uint64_t) 1L << 21); + carry6 = (s6 + (int64_t) (1L << 20)) >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry8 = (s8 + (int64_t) (1L << 20)) >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry10 = (s10 + (int64_t) (1L << 20)) >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + + carry1 = (s1 + (int64_t) (1L << 20)) >> 21; + s2 += carry1; + s1 -= carry1 * ((uint64_t) 1L << 21); + carry3 = (s3 + (int64_t) (1L << 20)) >> 21; + s4 += carry3; + s3 -= carry3 * ((uint64_t) 1L << 21); + carry5 = (s5 + (int64_t) (1L << 20)) >> 21; + s6 += carry5; + s5 -= carry5 * ((uint64_t) 1L << 21); + carry7 = (s7 + (int64_t) (1L << 20)) >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry9 = (s9 + (int64_t) (1L << 20)) >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry11 = (s11 + (int64_t) (1L << 20)) >> 21; + s12 += carry11; + s11 -= carry11 * ((uint64_t) 1L << 21); + + s0 += s12 * 666643; + s1 += s12 * 470296; + s2 += s12 * 654183; + s3 -= s12 * 997805; + s4 += s12 * 136657; + s5 -= s12 * 683901; + s12 = 0; + + carry0 = s0 >> 21; + s1 += carry0; + s0 -= carry0 * ((uint64_t) 1L << 21); + carry1 = s1 >> 21; + s2 += carry1; + s1 -= carry1 * ((uint64_t) 1L << 21); + carry2 = s2 >> 21; + s3 += carry2; + s2 -= carry2 * ((uint64_t) 1L << 21); + carry3 = s3 >> 21; + s4 += carry3; + s3 -= carry3 * ((uint64_t) 1L << 21); + carry4 = s4 >> 21; + s5 += carry4; + s4 -= carry4 * ((uint64_t) 1L << 21); + carry5 = s5 >> 21; + s6 += carry5; + s5 -= carry5 * ((uint64_t) 1L << 21); + carry6 = s6 >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry7 = s7 >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry8 = s8 >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry9 = s9 >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry10 = s10 >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + carry11 = s11 >> 21; + s12 += carry11; + s11 -= carry11 * ((uint64_t) 1L << 21); + + s0 += s12 * 666643; + s1 += s12 * 470296; + s2 += s12 * 654183; + s3 -= s12 * 997805; + s4 += s12 * 136657; + s5 -= s12 * 683901; + + carry0 = s0 >> 21; + s1 += carry0; + s0 -= carry0 * ((uint64_t) 1L << 21); + carry1 = s1 >> 21; + s2 += carry1; + s1 -= carry1 * ((uint64_t) 1L << 21); + carry2 = s2 >> 21; + s3 += carry2; + s2 -= carry2 * ((uint64_t) 1L << 21); + carry3 = s3 >> 21; + s4 += carry3; + s3 -= carry3 * ((uint64_t) 1L << 21); + carry4 = s4 >> 21; + s5 += carry4; + s4 -= carry4 * ((uint64_t) 1L << 21); + carry5 = s5 >> 21; + s6 += carry5; + s5 -= carry5 * ((uint64_t) 1L << 21); + carry6 = s6 >> 21; + s7 += carry6; + s6 -= carry6 * ((uint64_t) 1L << 21); + carry7 = s7 >> 21; + s8 += carry7; + s7 -= carry7 * ((uint64_t) 1L << 21); + carry8 = s8 >> 21; + s9 += carry8; + s8 -= carry8 * ((uint64_t) 1L << 21); + carry9 = s9 >> 21; + s10 += carry9; + s9 -= carry9 * ((uint64_t) 1L << 21); + carry10 = s10 >> 21; + s11 += carry10; + s10 -= carry10 * ((uint64_t) 1L << 21); + + s[0] = s0 >> 0; + s[1] = s0 >> 8; + s[2] = (s0 >> 16) | (s1 * ((uint64_t) 1 << 5)); + s[3] = s1 >> 3; + s[4] = s1 >> 11; + s[5] = (s1 >> 19) | (s2 * ((uint64_t) 1 << 2)); + s[6] = s2 >> 6; + s[7] = (s2 >> 14) | (s3 * ((uint64_t) 1 << 7)); + s[8] = s3 >> 1; + s[9] = s3 >> 9; + s[10] = (s3 >> 17) | (s4 * ((uint64_t) 1 << 4)); + s[11] = s4 >> 4; + s[12] = s4 >> 12; + s[13] = (s4 >> 20) | (s5 * ((uint64_t) 1 << 1)); + s[14] = s5 >> 7; + s[15] = (s5 >> 15) | (s6 * ((uint64_t) 1 << 6)); + s[16] = s6 >> 2; + s[17] = s6 >> 10; + s[18] = (s6 >> 18) | (s7 * ((uint64_t) 1 << 3)); + s[19] = s7 >> 5; + s[20] = s7 >> 13; + s[21] = s8 >> 0; + s[22] = s8 >> 8; + s[23] = (s8 >> 16) | (s9 * ((uint64_t) 1 << 5)); + s[24] = s9 >> 3; + s[25] = s9 >> 11; + s[26] = (s9 >> 19) | (s10 * ((uint64_t) 1 << 2)); + s[27] = s10 >> 6; + s[28] = (s10 >> 14) | (s11 * ((uint64_t) 1 << 7)); + s[29] = s11 >> 1; + s[30] = s11 >> 9; + s[31] = s11 >> 17; +} + +int +sc25519_is_canonical(const unsigned char s[32]) +{ + /* 2^252+27742317777372353535851937790883648493 */ + static const unsigned char L[32] = { + 0xed, 0xd3, 0xf5, 0x5c, 0x1a, 0x63, 0x12, 0x58, 0xd6, 0x9c, 0xf7, + 0xa2, 0xde, 0xf9, 0xde, 0x14, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10 + }; + unsigned char c = 0; + unsigned char n = 1; + unsigned int i = 32; + + do { + i--; + c |= ((s[i] - L[i]) >> 8) & n; + n &= ((s[i] ^ L[i]) - 1) >> 8; + } while (i != 0); + + return (c != 0); +} + +static void +chi25519(fe25519 out, const fe25519 z) +{ + fe25519 t0, t1, t2, t3; + int i; + + fe25519_sq(t0, z); + fe25519_mul(t1, t0, z); + fe25519_sq(t0, t1); + fe25519_sq(t2, t0); + fe25519_sq(t2, t2); + fe25519_mul(t2, t2, t0); + fe25519_mul(t1, t2, z); + fe25519_sq(t2, t1); + + for (i = 1; i < 5; i++) { + fe25519_sq(t2, t2); + } + fe25519_mul(t1, t2, t1); + fe25519_sq(t2, t1); + for (i = 1; i < 10; i++) { + fe25519_sq(t2, t2); + } + fe25519_mul(t2, t2, t1); + fe25519_sq(t3, t2); + for (i = 1; i < 20; i++) { + fe25519_sq(t3, t3); + } + fe25519_mul(t2, t3, t2); + fe25519_sq(t2, t2); + for (i = 1; i < 10; i++) { + fe25519_sq(t2, t2); + } + fe25519_mul(t1, t2, t1); + fe25519_sq(t2, t1); + for (i = 1; i < 50; i++) { + fe25519_sq(t2, t2); + } + fe25519_mul(t2, t2, t1); + fe25519_sq(t3, t2); + for (i = 1; i < 100; i++) { + fe25519_sq(t3, t3); + } + fe25519_mul(t2, t3, t2); + fe25519_sq(t2, t2); + for (i = 1; i < 50; i++) { + fe25519_sq(t2, t2); + } + fe25519_mul(t1, t2, t1); + fe25519_sq(t1, t1); + for (i = 1; i < 4; i++) { + fe25519_sq(t1, t1); + } + fe25519_mul(out, t1, t0); +} + +void +ge25519_from_uniform(unsigned char s[32], const unsigned char r[32]) +{ + fe25519 e; + fe25519 negx; + fe25519 rr2; + fe25519 x, x2, x3; + ge25519_p3 p3; + ge25519_p1p1 p1; + ge25519_p2 p2; + unsigned int e_is_minus_1; + unsigned char x_sign; + + memcpy(s, r, 32); + x_sign = s[31] & 0x80; + s[31] &= 0x7f; + + fe25519_frombytes(rr2, s); + + /* elligator */ + fe25519_sq2(rr2, rr2); + rr2[0]++; + fe25519_invert(rr2, rr2); + fe25519_mul(x, curve25519_A, rr2); + fe25519_neg(x, x); + + fe25519_sq(x2, x); + fe25519_mul(x3, x, x2); + fe25519_add(e, x3, x); + fe25519_mul(x2, x2, curve25519_A); + fe25519_add(e, x2, e); + + chi25519(e, e); + + fe25519_tobytes(s, e); + e_is_minus_1 = s[1] & 1; + fe25519_neg(negx, x); + fe25519_cmov(x, negx, e_is_minus_1); + fe25519_0(x2); + fe25519_cmov(x2, curve25519_A, e_is_minus_1); + fe25519_sub(x, x, x2); + + /* yed = (x-1)/(x+1) */ + { + fe25519 one; + fe25519 x_plus_one; + fe25519 x_plus_one_inv; + fe25519 x_minus_one; + fe25519 yed; + + fe25519_1(one); + fe25519_add(x_plus_one, x, one); + fe25519_sub(x_minus_one, x, one); + fe25519_invert(x_plus_one_inv, x_plus_one); + fe25519_mul(yed, x_minus_one, x_plus_one_inv); + fe25519_tobytes(s, yed); + } + + /* recover x */ + s[31] |= x_sign; + if (ge25519_frombytes(&p3, s) != 0) { + abort(); /* LCOV_EXCL_LINE */ + } + + /* multiply by the cofactor */ + ge25519_p3_dbl(&p1, &p3); + ge25519_p1p1_to_p2(&p2, &p1); + ge25519_p2_dbl(&p1, &p2); + ge25519_p1p1_to_p2(&p2, &p1); + ge25519_p2_dbl(&p1, &p2); + ge25519_p1p1_to_p3(&p3, &p1); + + ge25519_p3_tobytes(s, &p3); +} diff --git a/sodium/sodium_ref10_keypair.c b/sodium/sodium_ref10_keypair.c new file mode 100644 index 00000000..65225291 --- /dev/null +++ b/sodium/sodium_ref10_keypair.c @@ -0,0 +1,83 @@ + +#include + +#include "sodium_crypto_hash_sha512.h" +#include "sodium_crypto_scalarmult_curve25519.h" +#include "sodium_crypto_sign_ed25519.h" +#include "sodium_ref10_sign_ed25519.h" +#include "sodium_private_ed25519_ref10.h" +#include "sodium_randombytes.h" +#include "sodium_utils.h" + +int +crypto_sign_ed25519_seed_keypair(unsigned char *pk, unsigned char *sk, + const unsigned char *seed) +{ + ge25519_p3 A; + + crypto_hash_sha512(sk, seed, 32); + sk[0] &= 248; + sk[31] &= 127; + sk[31] |= 64; + + ge25519_scalarmult_base(&A, sk); + ge25519_p3_tobytes(pk, &A); + + memmove(sk, seed, 32); + memmove(sk + 32, pk, 32); + + return 0; +} + +int +crypto_sign_ed25519_keypair(unsigned char *pk, unsigned char *sk) +{ + unsigned char seed[32]; + int ret; + + randombytes_buf(seed, sizeof seed); + ret = crypto_sign_ed25519_seed_keypair(pk, sk, seed); + sodium_memzero(seed, sizeof seed); + + return ret; +} + +int +crypto_sign_ed25519_pk_to_curve25519(unsigned char *curve25519_pk, + const unsigned char *ed25519_pk) +{ + ge25519_p3 A; + fe25519 x; + fe25519 one_minus_y; + + if (ge25519_has_small_order(ed25519_pk) != 0 || + ge25519_frombytes_negate_vartime(&A, ed25519_pk) != 0 || + ge25519_is_on_main_subgroup(&A) == 0) { + return -1; + } + fe25519_1(one_minus_y); + fe25519_sub(one_minus_y, one_minus_y, A.Y); + fe25519_1(x); + fe25519_add(x, x, A.Y); + fe25519_invert(one_minus_y, one_minus_y); + fe25519_mul(x, x, one_minus_y); + fe25519_tobytes(curve25519_pk, x); + + return 0; +} + +int +crypto_sign_ed25519_sk_to_curve25519(unsigned char *curve25519_sk, + const unsigned char *ed25519_sk) +{ + unsigned char h[crypto_hash_sha512_BYTES]; + + crypto_hash_sha512(h, ed25519_sk, 32); + h[0] &= 248; + h[31] &= 127; + h[31] |= 64; + memcpy(curve25519_sk, h, crypto_scalarmult_curve25519_BYTES); + sodium_memzero(h, sizeof h); + + return 0; +} diff --git a/sodium/sodium_ref10_open.c b/sodium/sodium_ref10_open.c new file mode 100644 index 00000000..f8397c2a --- /dev/null +++ b/sodium/sodium_ref10_open.c @@ -0,0 +1,96 @@ + +#include +#include +#include + +#include "sodium_crypto_hash_sha512.h" +#include "sodium_crypto_sign_ed25519.h" +#include "sodium_crypto_verify_32.h" +#include "sodium_ref10_sign_ed25519.h" +#include "sodium_private_ed25519_ref10.h" +#include "sodium_utils.h" + +int +_crypto_sign_ed25519_verify_detached(const unsigned char *sig, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *pk, + int prehashed) +{ + crypto_hash_sha512_state hs; + unsigned char h[64]; + unsigned char rcheck[32]; + ge25519_p3 A; + ge25519_p2 R; + +#ifdef ED25519_COMPAT + if (sig[63] & 224) { + return -1; + } +#else + if (sc25519_is_canonical(sig + 32) == 0 || + ge25519_has_small_order(sig) != 0) { + return -1; + } + if (ge25519_is_canonical(pk) == 0 || + ge25519_has_small_order(pk) != 0) { + return -1; + } +#endif + if (ge25519_frombytes_negate_vartime(&A, pk) != 0) { + return -1; + } + _crypto_sign_ed25519_ref10_hinit(&hs, prehashed); + crypto_hash_sha512_update(&hs, sig, 32); + crypto_hash_sha512_update(&hs, pk, 32); + crypto_hash_sha512_update(&hs, m, mlen); + crypto_hash_sha512_final(&hs, h); + sc25519_reduce(h); + + ge25519_double_scalarmult_vartime(&R, h, &A, sig + 32); + ge25519_tobytes(rcheck, &R); + + return crypto_verify_32(rcheck, sig) | (-(rcheck == sig)) | + sodium_memcmp(sig, rcheck, 32); +} + +int +crypto_sign_ed25519_verify_detached(const unsigned char *sig, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *pk) +{ + return _crypto_sign_ed25519_verify_detached(sig, m, mlen, pk, 0); +} + +int +crypto_sign_ed25519_open(unsigned char *m, unsigned long long *mlen_p, + const unsigned char *sm, unsigned long long smlen, + const unsigned char *pk) +{ + unsigned long long mlen; + + if (smlen < 64 || smlen - 64 > crypto_sign_ed25519_MESSAGEBYTES_MAX) { + goto badsig; + } + mlen = smlen - 64; + if (crypto_sign_ed25519_verify_detached(sm, sm + 64, mlen, pk) != 0) { + if (m != NULL) { + memset(m, 0, mlen); + } + goto badsig; + } + if (mlen_p != NULL) { + *mlen_p = mlen; + } + if (m != NULL) { + memmove(m, sm + 64, mlen); + } + return 0; + +badsig: + if (mlen_p != NULL) { + *mlen_p = 0; + } + return -1; +} diff --git a/sodium/sodium_ref10_scalarmult_ed25519_ref10.c b/sodium/sodium_ref10_scalarmult_ed25519_ref10.c new file mode 100644 index 00000000..de21ccf5 --- /dev/null +++ b/sodium/sodium_ref10_scalarmult_ed25519_ref10.c @@ -0,0 +1,118 @@ + +#include + +#include "sodium_crypto_scalarmult_ed25519.h" +#include "sodium_private_ed25519_ref10.h" +#include "sodium_utils.h" + +static int +_crypto_scalarmult_ed25519_is_inf(const unsigned char s[32]) +{ + unsigned char c; + unsigned int i; + + c = s[0] ^ 0x01; + for (i = 1; i < 31; i++) { + c |= s[i]; + } + c |= s[31] & 0x7f; + + return ((((unsigned int) c) - 1U) >> 8) & 1; +} + +static inline void +_crypto_scalarmult_ed25519_clamp(unsigned char k[32]) +{ + k[0] &= 248; + k[31] &= 127; + k[31] |= 64; +} + +static int +_crypto_scalarmult_ed25519(unsigned char *q, const unsigned char *n, + const unsigned char *p, const int clamp) +{ + unsigned char *t = q; + ge25519_p3 Q; + ge25519_p3 P; + unsigned int i; + + if (ge25519_is_canonical(p) == 0 || ge25519_has_small_order(p) != 0 || + ge25519_frombytes(&P, p) != 0 || ge25519_is_on_main_subgroup(&P) == 0) { + return -1; + } + for (i = 0; i < 32; ++i) { + t[i] = n[i]; + } + if (clamp != 0) { + _crypto_scalarmult_ed25519_clamp(t); + } + ge25519_scalarmult(&Q, t, &P); + ge25519_p3_tobytes(q, &Q); + if (_crypto_scalarmult_ed25519_is_inf(q) != 0 || sodium_is_zero(n, 32)) { + return -1; + } + return 0; +} + +int +crypto_scalarmult_ed25519(unsigned char *q, const unsigned char *n, + const unsigned char *p) +{ + return _crypto_scalarmult_ed25519(q, n, p, 1); +} + +int +crypto_scalarmult_ed25519_noclamp(unsigned char *q, const unsigned char *n, + const unsigned char *p) +{ + return _crypto_scalarmult_ed25519(q, n, p, 0); +} + +static int +_crypto_scalarmult_ed25519_base(unsigned char *q, + const unsigned char *n, const int clamp) +{ + unsigned char *t = q; + ge25519_p3 Q; + unsigned int i; + + for (i = 0; i < 32; ++i) { + t[i] = n[i]; + } + if (clamp != 0) { + _crypto_scalarmult_ed25519_clamp(t); + } + ge25519_scalarmult_base(&Q, t); + ge25519_p3_tobytes(q, &Q); + if (_crypto_scalarmult_ed25519_is_inf(q) != 0 || sodium_is_zero(n, 32)) { + return -1; + } + return 0; +} + +int +crypto_scalarmult_ed25519_base(unsigned char *q, + const unsigned char *n) +{ + return _crypto_scalarmult_ed25519_base(q, n, 1); +} + +int +crypto_scalarmult_ed25519_base_noclamp(unsigned char *q, + const unsigned char *n) +{ + return _crypto_scalarmult_ed25519_base(q, n, 0); +} + +size_t +crypto_scalarmult_ed25519_bytes(void) +{ + return crypto_scalarmult_ed25519_BYTES; +} + +size_t +crypto_scalarmult_ed25519_scalarbytes(void) +{ + return crypto_scalarmult_ed25519_SCALARBYTES; +} diff --git a/sodium/sodium_ref10_sign.c b/sodium/sodium_ref10_sign.c new file mode 100644 index 00000000..6ac2b49f --- /dev/null +++ b/sodium/sodium_ref10_sign.c @@ -0,0 +1,143 @@ + +#include + +#include "sodium_crypto_hash_sha512.h" +#include "sodium_crypto_sign_ed25519.h" +#include "sodium_ref10_sign_ed25519.h" +#include "sodium_private_ed25519_ref10.h" +#include "sodium_randombytes.h" +#include "sodium_utils.h" + +void +_crypto_sign_ed25519_ref10_hinit(crypto_hash_sha512_state *hs, int prehashed) +{ + static const unsigned char DOM2PREFIX[32 + 2] = { + 'S', 'i', 'g', 'E', 'd', '2', '5', '5', '1', '9', ' ', + 'n', 'o', ' ', + 'E', 'd', '2', '5', '5', '1', '9', ' ', + 'c', 'o', 'l', 'l', 'i', 's', 'i', 'o', 'n', 's', 1, 0 + }; + + crypto_hash_sha512_init(hs); + if (prehashed) { + crypto_hash_sha512_update(hs, DOM2PREFIX, sizeof DOM2PREFIX); + } +} + +static inline void +_crypto_sign_ed25519_clamp(unsigned char k[32]) +{ + k[0] &= 248; + k[31] &= 127; + k[31] |= 64; +} + +#ifdef ED25519_NONDETERMINISTIC +/* r = hash(B || empty_labelset || Z || pad1 || k || pad2 || empty_labelset || K || extra || M) (mod q) */ +static void +_crypto_sign_ed25519_synthetic_r_hv(crypto_hash_sha512_state *hs, + unsigned char Z[32], + const unsigned char sk[64]) +{ + static const unsigned char B[32] = { + 0x58, 0x66, 0x66, 0x66, 0x66, 0x66, 0x66, 0x66, + 0x66, 0x66, 0x66, 0x66, 0x66, 0x66, 0x66, 0x66, + 0x66, 0x66, 0x66, 0x66, 0x66, 0x66, 0x66, 0x66, + 0x66, 0x66, 0x66, 0x66, 0x66, 0x66, 0x66, 0x66, + }; + static const unsigned char zeros[128] = { 0x00 }; + static const unsigned char empty_labelset[3] = { 0x02, 0x00, 0x00 }; + + crypto_hash_sha512_update(hs, B, 32); + crypto_hash_sha512_update(hs, empty_labelset, 3); + randombytes_buf(Z, 32); + crypto_hash_sha512_update(hs, Z, 32); + crypto_hash_sha512_update(hs, zeros, 128 - (32 + 3 + 32) % 128); + crypto_hash_sha512_update(hs, sk, 32); + crypto_hash_sha512_update(hs, zeros, 128 - 32 % 128); + crypto_hash_sha512_update(hs, empty_labelset, 3); + crypto_hash_sha512_update(hs, sk + 32, 32); + /* empty extra */ +} +#endif + +int +_crypto_sign_ed25519_detached(unsigned char *sig, unsigned long long *siglen_p, + const unsigned char *m, unsigned long long mlen, + const unsigned char *sk, int prehashed) +{ + crypto_hash_sha512_state hs; + unsigned char az[64]; + unsigned char nonce[64]; + unsigned char hram[64]; + ge25519_p3 R; + + _crypto_sign_ed25519_ref10_hinit(&hs, prehashed); + + crypto_hash_sha512(az, sk, 32); +#ifdef ED25519_NONDETERMINISTIC + _crypto_sign_ed25519_synthetic_r_hv(&hs, nonce, az); +#else + crypto_hash_sha512_update(&hs, az + 32, 32); +#endif + + crypto_hash_sha512_update(&hs, m, mlen); + crypto_hash_sha512_final(&hs, nonce); + + memmove(sig + 32, sk + 32, 32); + + sc25519_reduce(nonce); + ge25519_scalarmult_base(&R, nonce); + ge25519_p3_tobytes(sig, &R); + + _crypto_sign_ed25519_ref10_hinit(&hs, prehashed); + crypto_hash_sha512_update(&hs, sig, 64); + crypto_hash_sha512_update(&hs, m, mlen); + crypto_hash_sha512_final(&hs, hram); + + sc25519_reduce(hram); + _crypto_sign_ed25519_clamp(az); + sc25519_muladd(sig + 32, hram, az, nonce); + + sodium_memzero(az, sizeof az); + sodium_memzero(nonce, sizeof nonce); + + if (siglen_p != NULL) { + *siglen_p = 64U; + } + return 0; +} + +int +crypto_sign_ed25519_detached(unsigned char *sig, unsigned long long *siglen_p, + const unsigned char *m, unsigned long long mlen, + const unsigned char *sk) +{ + return _crypto_sign_ed25519_detached(sig, siglen_p, m, mlen, sk, 0); +} + +int +crypto_sign_ed25519(unsigned char *sm, unsigned long long *smlen_p, + const unsigned char *m, unsigned long long mlen, + const unsigned char *sk) +{ + unsigned long long siglen; + + memmove(sm + crypto_sign_ed25519_BYTES, m, mlen); + /* LCOV_EXCL_START */ + if (crypto_sign_ed25519_detached( + sm, &siglen, sm + crypto_sign_ed25519_BYTES, mlen, sk) != 0 || + siglen != crypto_sign_ed25519_BYTES) { + if (smlen_p != NULL) { + *smlen_p = 0; + } + memset(sm, 0, mlen + crypto_sign_ed25519_BYTES); + return -1; + } + /* LCOV_EXCL_STOP */ + + if (smlen_p != NULL) { + *smlen_p = mlen + siglen; + } + return 0; +} diff --git a/sodium/sodium_ref10_sign_ed25519.h b/sodium/sodium_ref10_sign_ed25519.h new file mode 100644 index 00000000..29f45a85 --- /dev/null +++ b/sodium/sodium_ref10_sign_ed25519.h @@ -0,0 +1,18 @@ +#ifndef sign_ed25519_ref10_H +#define sign_ed25519_ref10_H + +void _crypto_sign_ed25519_ref10_hinit(crypto_hash_sha512_state *hs, + int prehashed); + +int _crypto_sign_ed25519_detached(unsigned char *sig, + unsigned long long *siglen_p, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *sk, int prehashed); + +int _crypto_sign_ed25519_verify_detached(const unsigned char *sig, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *pk, + int prehashed); +#endif diff --git a/sodium/sodium_ref10_x25519.c b/sodium/sodium_ref10_x25519.c new file mode 100644 index 00000000..3eb7e193 --- /dev/null +++ b/sodium/sodium_ref10_x25519.c @@ -0,0 +1,180 @@ + +#include +#include + +#include "sodium_scalarmult_curve25519.h" +#include "sodium_export.h" +#include "sodium_private_ed25519_ref10.h" +#include "sodium_utils.h" +#include "sodium_ref10_x25519.h" + +/* + * Reject small order points early to mitigate the implications of + * unexpected optimizations that would affect the ref10 code. + * See https://eprint.iacr.org/2017/806.pdf for reference. + */ +static int +has_small_order(const unsigned char s[32]) +{ + CRYPTO_ALIGN(16) + static const unsigned char blacklist[][32] = { + /* 0 (order 4) */ + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, + /* 1 (order 1) */ + { 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, + /* 325606250916557431795983626356110631294008115727848805560023387167927233504 + (order 8) */ + { 0xe0, 0xeb, 0x7a, 0x7c, 0x3b, 0x41, 0xb8, 0xae, 0x16, 0x56, 0xe3, + 0xfa, 0xf1, 0x9f, 0xc4, 0x6a, 0xda, 0x09, 0x8d, 0xeb, 0x9c, 0x32, + 0xb1, 0xfd, 0x86, 0x62, 0x05, 0x16, 0x5f, 0x49, 0xb8, 0x00 }, + /* 39382357235489614581723060781553021112529911719440698176882885853963445705823 + (order 8) */ + { 0x5f, 0x9c, 0x95, 0xbc, 0xa3, 0x50, 0x8c, 0x24, 0xb1, 0xd0, 0xb1, + 0x55, 0x9c, 0x83, 0xef, 0x5b, 0x04, 0x44, 0x5c, 0xc4, 0x58, 0x1c, + 0x8e, 0x86, 0xd8, 0x22, 0x4e, 0xdd, 0xd0, 0x9f, 0x11, 0x57 }, + /* p-1 (order 2) */ + { 0xec, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x7f }, + /* p (=0, order 4) */ + { 0xed, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x7f }, + /* p+1 (=1, order 1) */ + { 0xee, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, + 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x7f } + }; + unsigned char c[7] = { 0 }; + unsigned int k; + size_t i, j; + + COMPILER_ASSERT(7 == sizeof blacklist / sizeof blacklist[0]); + for (j = 0; j < 31; j++) { + for (i = 0; i < sizeof blacklist / sizeof blacklist[0]; i++) { + c[i] |= s[j] ^ blacklist[i][j]; + } + } + for (i = 0; i < sizeof blacklist / sizeof blacklist[0]; i++) { + c[i] |= (s[j] & 0x7f) ^ blacklist[i][j]; + } + k = 0; + for (i = 0; i < sizeof blacklist / sizeof blacklist[0]; i++) { + k |= (c[i] - 1); + } + return (int) ((k >> 8) & 1); +} + +static int +crypto_scalarmult_curve25519_ref10(unsigned char *q, + const unsigned char *n, + const unsigned char *p) +{ + unsigned char *t = q; + unsigned int i; + fe25519 x1; + fe25519 x2; + fe25519 z2; + fe25519 x3; + fe25519 z3; + fe25519 tmp0; + fe25519 tmp1; + int pos; + unsigned int swap; + unsigned int b; + + if (has_small_order(p)) { + return -1; + } + for (i = 0; i < 32; i++) { + t[i] = n[i]; + } + t[0] &= 248; + t[31] &= 127; + t[31] |= 64; + fe25519_frombytes(x1, p); + fe25519_1(x2); + fe25519_0(z2); + fe25519_copy(x3, x1); + fe25519_1(z3); + + swap = 0; + for (pos = 254; pos >= 0; --pos) { + b = t[pos / 8] >> (pos & 7); + b &= 1; + swap ^= b; + fe25519_cswap(x2, x3, swap); + fe25519_cswap(z2, z3, swap); + swap = b; + fe25519_sub(tmp0, x3, z3); + fe25519_sub(tmp1, x2, z2); + fe25519_add(x2, x2, z2); + fe25519_add(z2, x3, z3); + fe25519_mul(z3, tmp0, x2); + fe25519_mul(z2, z2, tmp1); + fe25519_sq(tmp0, tmp1); + fe25519_sq(tmp1, x2); + fe25519_add(x3, z3, z2); + fe25519_sub(z2, z3, z2); + fe25519_mul(x2, tmp1, tmp0); + fe25519_sub(tmp1, tmp1, tmp0); + fe25519_sq(z2, z2); + fe25519_scalar_product(z3, tmp1, 121666); + fe25519_sq(x3, x3); + fe25519_add(tmp0, tmp0, z3); + fe25519_mul(z3, x1, z2); + fe25519_mul(z2, tmp1, tmp0); + } + fe25519_cswap(x2, x3, swap); + fe25519_cswap(z2, z3, swap); + + fe25519_invert(z2, z2); + fe25519_mul(x2, x2, z2); + fe25519_tobytes(q, x2); + + return 0; +} + +static void +edwards_to_montgomery(fe25519 montgomeryX, const fe25519 edwardsY, const fe25519 edwardsZ) +{ + fe25519 tempX; + fe25519 tempZ; + + fe25519_add(tempX, edwardsZ, edwardsY); + fe25519_sub(tempZ, edwardsZ, edwardsY); + fe25519_invert(tempZ, tempZ); + fe25519_mul(montgomeryX, tempX, tempZ); +} + +static int +crypto_scalarmult_curve25519_ref10_base(unsigned char *q, + const unsigned char *n) +{ + unsigned char *t = q; + ge25519_p3 A; + fe25519 pk; + unsigned int i; + + for (i = 0; i < 32; i++) { + t[i] = n[i]; + } + t[0] &= 248; + t[31] &= 127; + t[31] |= 64; + ge25519_scalarmult_base(&A, t); + edwards_to_montgomery(pk, A.Y, A.Z); + fe25519_tobytes(q, pk); + + return 0; +} + +struct crypto_scalarmult_curve25519_implementation + crypto_scalarmult_curve25519_ref10_implementation = { + SODIUM_C99(.mult =) crypto_scalarmult_curve25519_ref10, + SODIUM_C99(.mult_base =) crypto_scalarmult_curve25519_ref10_base + }; diff --git a/sodium/sodium_ref10_x25519.h b/sodium/sodium_ref10_x25519.h new file mode 100644 index 00000000..b2d55dcf --- /dev/null +++ b/sodium/sodium_ref10_x25519.h @@ -0,0 +1,10 @@ +#ifndef x25519_ref10_H +#define x25519_ref10_H + +#include "sodium_crypto_scalarmult_curve25519.h" +#include "sodium_scalarmult_curve25519.h" + +extern struct crypto_scalarmult_curve25519_implementation + crypto_scalarmult_curve25519_ref10_implementation; + +#endif diff --git a/sodium/sodium_runtime.c b/sodium/sodium_runtime.c new file mode 100644 index 00000000..7d57e9a2 --- /dev/null +++ b/sodium/sodium_runtime.c @@ -0,0 +1,319 @@ +#include +#include +#ifdef HAVE_ANDROID_GETCPUFEATURES +# include +#endif + +#include "sodium_private_common.h" +#include "sodium_runtime.h" +#include + +typedef struct CPUFeatures_ { + int initialized; + int has_neon; + int has_sse2; + int has_sse3; + int has_ssse3; + int has_sse41; + int has_avx; + int has_avx2; + int has_avx512f; + int has_pclmul; + int has_aesni; + int has_rdrand; +} CPUFeatures; + +static CPUFeatures _cpu_features; + +#define CPUID_EBX_AVX2 0x00000020 +#define CPUID_EBX_AVX512F 0x00010000 + +#define CPUID_ECX_SSE3 0x00000001 +#define CPUID_ECX_PCLMUL 0x00000002 +#define CPUID_ECX_SSSE3 0x00000200 +#define CPUID_ECX_SSE41 0x00080000 +#define CPUID_ECX_AESNI 0x02000000 +#define CPUID_ECX_XSAVE 0x04000000 +#define CPUID_ECX_OSXSAVE 0x08000000 +#define CPUID_ECX_AVX 0x10000000 +#define CPUID_ECX_RDRAND 0x40000000 + +#define CPUID_EDX_SSE2 0x04000000 + +#define XCR0_SSE 0x00000002 +#define XCR0_AVX 0x00000004 +#define XCR0_OPMASK 0x00000020 +#define XCR0_ZMM_HI256 0x00000040 +#define XCR0_HI16_ZMM 0x00000080 + +static int +_sodium_runtime_arm_cpu_features(CPUFeatures * const cpu_features) +{ +#ifndef __arm__ + cpu_features->has_neon = 0; + return -1; +#else +# ifdef __APPLE__ +# ifdef __ARM_NEON__ + cpu_features->has_neon = 1; +# else + cpu_features->has_neon = 0; +# endif +# elif defined(HAVE_ANDROID_GETCPUFEATURES) && \ + defined(ANDROID_CPU_ARM_FEATURE_NEON) + cpu_features->has_neon = + (android_getCpuFeatures() & ANDROID_CPU_ARM_FEATURE_NEON) != 0x0; +# else + cpu_features->has_neon = 0; +# endif + return 0; +#endif +} + +static void +_cpuid(unsigned int cpu_info[4U], const unsigned int cpu_info_type) +{ +#if defined(_MSC_VER) && \ + (defined(_M_X64) || defined(_M_AMD64) || defined(_M_IX86)) + __cpuid((int *) cpu_info, cpu_info_type); +#elif defined(HAVE_CPUID) + cpu_info[0] = cpu_info[1] = cpu_info[2] = cpu_info[3] = 0; +# ifdef __i386__ + __asm__ __volatile__( + "pushfl; pushfl; " + "popl %0; " + "movl %0, %1; xorl %2, %0; " + "pushl %0; " + "popfl; pushfl; popl %0; popfl" + : "=&r"(cpu_info[0]), "=&r"(cpu_info[1]) + : "i"(0x200000)); + if (((cpu_info[0] ^ cpu_info[1]) & 0x200000) == 0x0) { + return; /* LCOV_EXCL_LINE */ + } +# endif +# ifdef __i386__ + __asm__ __volatile__("xchgl %%ebx, %k1; cpuid; xchgl %%ebx, %k1" + : "=a"(cpu_info[0]), "=&r"(cpu_info[1]), + "=c"(cpu_info[2]), "=d"(cpu_info[3]) + : "0"(cpu_info_type), "2"(0U)); +# elif defined(__x86_64__) + __asm__ __volatile__("xchgq %%rbx, %q1; cpuid; xchgq %%rbx, %q1" + : "=a"(cpu_info[0]), "=&r"(cpu_info[1]), + "=c"(cpu_info[2]), "=d"(cpu_info[3]) + : "0"(cpu_info_type), "2"(0U)); +# else + __asm__ __volatile__("cpuid" + : "=a"(cpu_info[0]), "=b"(cpu_info[1]), + "=c"(cpu_info[2]), "=d"(cpu_info[3]) + : "0"(cpu_info_type), "2"(0U)); +# endif +#else + (void) cpu_info_type; + cpu_info[0] = cpu_info[1] = cpu_info[2] = cpu_info[3] = 0; +#endif +} + +static int +_sodium_runtime_intel_cpu_features(CPUFeatures * const cpu_features) +{ + unsigned int cpu_info[4]; + unsigned int id; + uint32_t xcr0 = 0U; + + _cpuid(cpu_info, 0x0); + if ((id = cpu_info[0]) == 0U) { + return -1; /* LCOV_EXCL_LINE */ + } + _cpuid(cpu_info, 0x00000001); +#ifdef HAVE_EMMINTRIN_H + cpu_features->has_sse2 = ((cpu_info[3] & CPUID_EDX_SSE2) != 0x0); +#else + cpu_features->has_sse2 = 0; +#endif + +#ifdef HAVE_PMMINTRIN_H + cpu_features->has_sse3 = ((cpu_info[2] & CPUID_ECX_SSE3) != 0x0); +#else + cpu_features->has_sse3 = 0; +#endif + +#ifdef HAVE_TMMINTRIN_H + cpu_features->has_ssse3 = ((cpu_info[2] & CPUID_ECX_SSSE3) != 0x0); +#else + cpu_features->has_ssse3 = 0; +#endif + +#ifdef HAVE_SMMINTRIN_H + cpu_features->has_sse41 = ((cpu_info[2] & CPUID_ECX_SSE41) != 0x0); +#else + cpu_features->has_sse41 = 0; +#endif + + cpu_features->has_avx = 0; + + (void) xcr0; +#ifdef HAVE_AVXINTRIN_H + if ((cpu_info[2] & (CPUID_ECX_AVX | CPUID_ECX_XSAVE | CPUID_ECX_OSXSAVE)) == + (CPUID_ECX_AVX | CPUID_ECX_XSAVE | CPUID_ECX_OSXSAVE)) { + xcr0 = 0U; +# if defined(HAVE__XGETBV) || \ + (defined(_MSC_VER) && defined(_XCR_XFEATURE_ENABLED_MASK) && _MSC_FULL_VER >= 160040219) + xcr0 = (uint32_t) _xgetbv(0); +# elif defined(_MSC_VER) && defined(_M_IX86) + /* + * Visual Studio documentation states that eax/ecx/edx don't need to + * be preserved in inline assembly code. But that doesn't seem to + * always hold true on Visual Studio 2010. + */ + __asm { + push eax + push ecx + push edx + xor ecx, ecx + _asm _emit 0x0f _asm _emit 0x01 _asm _emit 0xd0 + mov xcr0, eax + pop edx + pop ecx + pop eax + } +# elif defined(HAVE_AVX_ASM) + __asm__ __volatile__(".byte 0x0f, 0x01, 0xd0" /* XGETBV */ + : "=a"(xcr0) + : "c"((uint32_t) 0U) + : "%edx"); +# endif + if ((xcr0 & (XCR0_SSE | XCR0_AVX)) == (XCR0_SSE | XCR0_AVX)) { + cpu_features->has_avx = 1; + } + } +#endif + + cpu_features->has_avx2 = 0; +#ifdef HAVE_AVX2INTRIN_H + if (cpu_features->has_avx) { + unsigned int cpu_info7[4]; + + _cpuid(cpu_info7, 0x00000007); + cpu_features->has_avx2 = ((cpu_info7[1] & CPUID_EBX_AVX2) != 0x0); + } +#endif + + cpu_features->has_avx512f = 0; +#ifdef HAVE_AVX512FINTRIN_H + if (cpu_features->has_avx2) { + unsigned int cpu_info7[4]; + + _cpuid(cpu_info7, 0x00000007); + if ((cpu_info7[1] & CPUID_EBX_AVX512F) == CPUID_EBX_AVX512F && + (xcr0 & (XCR0_OPMASK | XCR0_ZMM_HI256 | XCR0_HI16_ZMM)) + == (XCR0_OPMASK | XCR0_ZMM_HI256 | XCR0_HI16_ZMM)) { + cpu_features->has_avx512f = 1; + } + } +#endif + +#ifdef HAVE_WMMINTRIN_H + cpu_features->has_pclmul = ((cpu_info[2] & CPUID_ECX_PCLMUL) != 0x0); + cpu_features->has_aesni = ((cpu_info[2] & CPUID_ECX_AESNI) != 0x0); +#else + cpu_features->has_pclmul = 0; + cpu_features->has_aesni = 0; +#endif + +#ifdef HAVE_RDRAND + cpu_features->has_rdrand = ((cpu_info[2] & CPUID_ECX_RDRAND) != 0x0); +#else + cpu_features->has_rdrand = 0; +#endif + + return 0; +} + +int +_sodium_runtime_get_cpu_features(void) +{ + int ret = -1; + + ret &= _sodium_runtime_arm_cpu_features(&_cpu_features); + ret &= _sodium_runtime_intel_cpu_features(&_cpu_features); + _cpu_features.initialized = 1; + + #if NETCODE_CRYPTO_LOGS + printf( "\nCPU features: " ); + if ( _cpu_features.has_sse2 ) printf( "sse2 " ); + if ( _cpu_features.has_ssse3 ) printf( "ssse3 " ); + if ( _cpu_features.has_sse41 ) printf( "sse41 " ); + if ( _cpu_features.has_avx ) printf( "avx " ); + if ( _cpu_features.has_avx2 ) printf( "avx2 " ); + if ( _cpu_features.has_avx512f ) printf( "avx512f " ); + printf( "\n\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + + return ret; +} + +int +sodium_runtime_has_neon(void) +{ + return _cpu_features.has_neon; +} + +int +sodium_runtime_has_sse2(void) +{ + return _cpu_features.has_sse2; +} + +int +sodium_runtime_has_sse3(void) +{ + return _cpu_features.has_sse3; +} + +int +sodium_runtime_has_ssse3(void) +{ + return _cpu_features.has_ssse3; +} + +int +sodium_runtime_has_sse41(void) +{ + return _cpu_features.has_sse41; +} + +int +sodium_runtime_has_avx(void) +{ + return _cpu_features.has_avx; +} + +int +sodium_runtime_has_avx2(void) +{ + return _cpu_features.has_avx2; +} + +int +sodium_runtime_has_avx512f(void) +{ + return _cpu_features.has_avx512f; +} + +int +sodium_runtime_has_pclmul(void) +{ + return _cpu_features.has_pclmul; +} + +int +sodium_runtime_has_aesni(void) +{ + return _cpu_features.has_aesni; +} + +int +sodium_runtime_has_rdrand(void) +{ + return _cpu_features.has_rdrand; +} diff --git a/windows/sodium/runtime.h b/sodium/sodium_runtime.h similarity index 79% rename from windows/sodium/runtime.h rename to sodium/sodium_runtime.h index 76859ea0..c1a30572 100644 --- a/windows/sodium/runtime.h +++ b/sodium/sodium_runtime.h @@ -2,39 +2,34 @@ #ifndef sodium_runtime_H #define sodium_runtime_H -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { #endif -SODIUM_EXPORT int sodium_runtime_has_neon(void); -SODIUM_EXPORT int sodium_runtime_has_sse2(void); -SODIUM_EXPORT int sodium_runtime_has_sse3(void); -SODIUM_EXPORT int sodium_runtime_has_ssse3(void); -SODIUM_EXPORT int sodium_runtime_has_sse41(void); -SODIUM_EXPORT int sodium_runtime_has_avx(void); -SODIUM_EXPORT int sodium_runtime_has_avx2(void); -SODIUM_EXPORT +int sodium_runtime_has_avx512f(void); + int sodium_runtime_has_pclmul(void); -SODIUM_EXPORT int sodium_runtime_has_aesni(void); +int sodium_runtime_has_rdrand(void); + /* ------------------------------------------------------------------------- */ int _sodium_runtime_get_cpu_features(void); diff --git a/sodium/sodium_salsa20-ref.c b/sodium/sodium_salsa20-ref.c new file mode 100644 index 00000000..1da26723 --- /dev/null +++ b/sodium/sodium_salsa20-ref.c @@ -0,0 +1,124 @@ +/* +version 20140420 +D. J. Bernstein +Public domain. +*/ + +#include + +#include "sodium_crypto_core_salsa20.h" +#include "sodium_crypto_stream_salsa20.h" +#include "sodium_utils.h" + +#include "sodium_stream_salsa20.h" +#include "sodium_salsa20-ref.h" + +#ifndef HAVE_AMD64_ASM + +static int +stream_ref(unsigned char *c, unsigned long long clen, const unsigned char *n, + const unsigned char *k) +{ + unsigned char in[16]; + unsigned char block[64]; + unsigned char kcopy[32]; + unsigned int i; + unsigned int u; + + if (!clen) { + return 0; + } + for (i = 0; i < 32; i++) { + kcopy[i] = k[i]; + } + for (i = 0; i < 8; i++) { + in[i] = n[i]; + } + for (i = 8; i < 16; i++) { + in[i] = 0; + } + while (clen >= 64) { + crypto_core_salsa20(c, in, kcopy, NULL); + u = 1; + for (i = 8; i < 16; i++) { + u += (unsigned int) in[i]; + in[i] = (unsigned char) u; + u >>= 8; + } + clen -= 64; + c += 64; + } + if (clen) { + crypto_core_salsa20(block, in, kcopy, NULL); + for (i = 0; i < (unsigned int) clen; i++) { + c[i] = block[i]; + } + } + sodium_memzero(block, sizeof block); + sodium_memzero(kcopy, sizeof kcopy); + + return 0; +} + +static int +stream_ref_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, uint64_t ic, + const unsigned char *k) +{ + unsigned char in[16]; + unsigned char block[64]; + unsigned char kcopy[32]; + unsigned int i; + unsigned int u; + + if (!mlen) { + return 0; + } + for (i = 0; i < 32; i++) { + kcopy[i] = k[i]; + } + for (i = 0; i < 8; i++) { + in[i] = n[i]; + } + for (i = 8; i < 16; i++) { + in[i] = (unsigned char) (ic & 0xff); + ic >>= 8; + } + while (mlen >= 64) { + crypto_core_salsa20(block, in, kcopy, NULL); + for (i = 0; i < 64; i++) { + c[i] = m[i] ^ block[i]; + } + u = 1; + for (i = 8; i < 16; i++) { + u += (unsigned int) in[i]; + in[i] = (unsigned char) u; + u >>= 8; + } + mlen -= 64; + c += 64; + m += 64; + } + if (mlen) { + crypto_core_salsa20(block, in, kcopy, NULL); + for (i = 0; i < (unsigned int) mlen; i++) { + c[i] = m[i] ^ block[i]; + } + } + sodium_memzero(block, sizeof block); + sodium_memzero(kcopy, sizeof kcopy); + + return 0; +} + +struct crypto_stream_salsa20_implementation + crypto_stream_salsa20_ref_implementation = { + SODIUM_C99(.stream =) stream_ref, + SODIUM_C99(.stream_xor_ic =) stream_ref_xor_ic, + }; + +#else + +int salsa20_ref_dummy = 0; + +#endif diff --git a/sodium/sodium_salsa20-ref.h b/sodium/sodium_salsa20-ref.h new file mode 100644 index 00000000..8035ed1f --- /dev/null +++ b/sodium/sodium_salsa20-ref.h @@ -0,0 +1,8 @@ + +#include + +#include "sodium_stream_salsa20.h" +#include "sodium_crypto_stream_salsa20.h" + +extern struct crypto_stream_salsa20_implementation + crypto_stream_salsa20_ref_implementation; diff --git a/sodium/sodium_salsa20-xmm6.c b/sodium/sodium_salsa20-xmm6.c new file mode 100644 index 00000000..c52686f2 --- /dev/null +++ b/sodium/sodium_salsa20-xmm6.c @@ -0,0 +1,33 @@ + +#include + +#include "sodium_utils.h" +#include "sodium_stream_salsa20.h" +#include "sodium_salsa20-xmm6.h" +#include "sodium_private_common.h" + +#ifdef HAVE_AMD64_ASM + +#ifdef __cplusplus +extern "C" { +#endif +extern int stream_salsa20_xmm6(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k); + +extern int stream_salsa20_xmm6_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, + uint64_t ic, const unsigned char *k); +#ifdef __cplusplus +} +#endif + +struct crypto_stream_salsa20_implementation + crypto_stream_salsa20_xmm6_implementation = { + SODIUM_C99(.stream =) stream_salsa20_xmm6, + SODIUM_C99(.stream_xor_ic =) stream_salsa20_xmm6_xor_ic, + }; + +#endif + +int salsa20_xmm6_link_warning_dummy = 0; diff --git a/sodium/sodium_salsa20-xmm6.h b/sodium/sodium_salsa20-xmm6.h new file mode 100644 index 00000000..19d6b1bf --- /dev/null +++ b/sodium/sodium_salsa20-xmm6.h @@ -0,0 +1,8 @@ + +#include + +#include "sodium_stream_salsa20.h" +#include "sodium_crypto_stream_salsa20.h" + +extern struct crypto_stream_salsa20_implementation + crypto_stream_salsa20_xmm6_implementation; diff --git a/sodium/sodium_salsa20-xmm6_asm.S b/sodium/sodium_salsa20-xmm6_asm.S new file mode 100644 index 00000000..0444928b --- /dev/null +++ b/sodium/sodium_salsa20-xmm6_asm.S @@ -0,0 +1,962 @@ +#ifdef NETCODE_X64 + +.text +.p2align 5 + +#ifdef ASM_HIDE_SYMBOL +ASM_HIDE_SYMBOL stream_salsa20_xmm6 +ASM_HIDE_SYMBOL _stream_salsa20_xmm6 +#endif +.globl stream_salsa20_xmm6 +.globl _stream_salsa20_xmm6 +#ifdef __ELF__ +.type stream_salsa20_xmm6, @function +.type _stream_salsa20_xmm6, @function +#endif +stream_salsa20_xmm6: +_stream_salsa20_xmm6: +mov %rsp,%r11 +and $31,%r11 +add $512,%r11 +sub %r11,%rsp +movq %r11,416(%rsp) +movq %r12,424(%rsp) +movq %r13,432(%rsp) +movq %r14,440(%rsp) +movq %r15,448(%rsp) +movq %rbx,456(%rsp) +movq %rbp,464(%rsp) +mov %rsi,%r9 +mov %rdi,%rdi +mov %rdi,%rsi +mov %rdx,%rdx +mov %rcx,%r10 +cmp $0,%r9 +jbe ._done +mov $0,%rax +mov %r9,%rcx +rep stosb +sub %r9,%rdi +movq $0,472(%rsp) +jmp ._start + +.text +.p2align 5 + +#ifdef ASM_HIDE_SYMBOL +ASM_HIDE_SYMBOL stream_salsa20_xmm6_xor_ic +ASM_HIDE_SYMBOL _stream_salsa20_xmm6_xor_ic +#endif +.globl stream_salsa20_xmm6_xor_ic +.globl _stream_salsa20_xmm6_xor_ic +#ifdef __ELF__ +.type stream_salsa20_xmm6_xor_ic, @function +.type _stream_salsa20_xmm6_xor_ic, @function +#endif +stream_salsa20_xmm6_xor_ic: +_stream_salsa20_xmm6_xor_ic: + +mov %rsp,%r11 +and $31,%r11 +add $512,%r11 +sub %r11,%rsp +movq %r11,416(%rsp) +movq %r12,424(%rsp) +movq %r13,432(%rsp) +movq %r14,440(%rsp) +movq %r15,448(%rsp) +movq %rbx,456(%rsp) +movq %rbp,464(%rsp) +mov %rdi,%rdi +mov %rsi,%rsi +mov %r9,%r10 +movq %r8,472(%rsp) +mov %rdx,%r9 +mov %rcx,%rdx +cmp $0,%r9 +jbe ._done + +._start: +movl 20(%r10),%ecx +movl 0(%r10),%r8d +movl 0(%rdx),%eax +movl 16(%r10),%r11d +movl %ecx,64(%rsp) +movl %r8d,4+64(%rsp) +movl %eax,8+64(%rsp) +movl %r11d,12+64(%rsp) +movl 24(%r10),%r8d +movl 4(%r10),%eax +movl 4(%rdx),%edx +movq 472(%rsp),%rcx +movl %ecx,80(%rsp) +movl %r8d,4+80(%rsp) +movl %eax,8+80(%rsp) +movl %edx,12+80(%rsp) +movl 12(%r10),%edx +shr $32,%rcx +movl 28(%r10),%r8d +movl 8(%r10),%eax +movl %edx,96(%rsp) +movl %ecx,4+96(%rsp) +movl %r8d,8+96(%rsp) +movl %eax,12+96(%rsp) +mov $1634760805,%rdx +mov $857760878,%rcx +mov $2036477234,%r8 +mov $1797285236,%rax +movl %edx,112(%rsp) +movl %ecx,4+112(%rsp) +movl %r8d,8+112(%rsp) +movl %eax,12+112(%rsp) +cmp $256,%r9 +jb ._bytesbetween1and255 +movdqa 112(%rsp),%xmm0 +pshufd $0x55,%xmm0,%xmm1 +pshufd $0xaa,%xmm0,%xmm2 +pshufd $0xff,%xmm0,%xmm3 +pshufd $0x00,%xmm0,%xmm0 +movdqa %xmm1,128(%rsp) +movdqa %xmm2,144(%rsp) +movdqa %xmm3,160(%rsp) +movdqa %xmm0,176(%rsp) +movdqa 64(%rsp),%xmm0 +pshufd $0xaa,%xmm0,%xmm1 +pshufd $0xff,%xmm0,%xmm2 +pshufd $0x00,%xmm0,%xmm3 +pshufd $0x55,%xmm0,%xmm0 +movdqa %xmm1,192(%rsp) +movdqa %xmm2,208(%rsp) +movdqa %xmm3,224(%rsp) +movdqa %xmm0,240(%rsp) +movdqa 80(%rsp),%xmm0 +pshufd $0xff,%xmm0,%xmm1 +pshufd $0x55,%xmm0,%xmm2 +pshufd $0xaa,%xmm0,%xmm0 +movdqa %xmm1,256(%rsp) +movdqa %xmm2,272(%rsp) +movdqa %xmm0,288(%rsp) +movdqa 96(%rsp),%xmm0 +pshufd $0x00,%xmm0,%xmm1 +pshufd $0xaa,%xmm0,%xmm2 +pshufd $0xff,%xmm0,%xmm0 +movdqa %xmm1,304(%rsp) +movdqa %xmm2,320(%rsp) +movdqa %xmm0,336(%rsp) + +.p2align 4 +._bytesatleast256: +movq 472(%rsp),%rdx +mov %rdx,%rcx +shr $32,%rcx +movl %edx,352(%rsp) +movl %ecx,368(%rsp) +add $1,%rdx +mov %rdx,%rcx +shr $32,%rcx +movl %edx,4+352(%rsp) +movl %ecx,4+368(%rsp) +add $1,%rdx +mov %rdx,%rcx +shr $32,%rcx +movl %edx,8+352(%rsp) +movl %ecx,8+368(%rsp) +add $1,%rdx +mov %rdx,%rcx +shr $32,%rcx +movl %edx,12+352(%rsp) +movl %ecx,12+368(%rsp) +add $1,%rdx +mov %rdx,%rcx +shr $32,%rcx +movl %edx,80(%rsp) +movl %ecx,4+96(%rsp) +movq %rdx,472(%rsp) +movq %r9,480(%rsp) +mov $20,%rdx +movdqa 128(%rsp),%xmm0 +movdqa 144(%rsp),%xmm1 +movdqa 160(%rsp),%xmm2 +movdqa 320(%rsp),%xmm3 +movdqa 336(%rsp),%xmm4 +movdqa 192(%rsp),%xmm5 +movdqa 208(%rsp),%xmm6 +movdqa 240(%rsp),%xmm7 +movdqa 256(%rsp),%xmm8 +movdqa 272(%rsp),%xmm9 +movdqa 288(%rsp),%xmm10 +movdqa 368(%rsp),%xmm11 +movdqa 176(%rsp),%xmm12 +movdqa 224(%rsp),%xmm13 +movdqa 304(%rsp),%xmm14 +movdqa 352(%rsp),%xmm15 + +.p2align 4 +._mainloop1: +movdqa %xmm1,384(%rsp) +movdqa %xmm2,400(%rsp) +movdqa %xmm13,%xmm1 +paddd %xmm12,%xmm1 +movdqa %xmm1,%xmm2 +pslld $7,%xmm1 +pxor %xmm1,%xmm14 +psrld $25,%xmm2 +pxor %xmm2,%xmm14 +movdqa %xmm7,%xmm1 +paddd %xmm0,%xmm1 +movdqa %xmm1,%xmm2 +pslld $7,%xmm1 +pxor %xmm1,%xmm11 +psrld $25,%xmm2 +pxor %xmm2,%xmm11 +movdqa %xmm12,%xmm1 +paddd %xmm14,%xmm1 +movdqa %xmm1,%xmm2 +pslld $9,%xmm1 +pxor %xmm1,%xmm15 +psrld $23,%xmm2 +pxor %xmm2,%xmm15 +movdqa %xmm0,%xmm1 +paddd %xmm11,%xmm1 +movdqa %xmm1,%xmm2 +pslld $9,%xmm1 +pxor %xmm1,%xmm9 +psrld $23,%xmm2 +pxor %xmm2,%xmm9 +movdqa %xmm14,%xmm1 +paddd %xmm15,%xmm1 +movdqa %xmm1,%xmm2 +pslld $13,%xmm1 +pxor %xmm1,%xmm13 +psrld $19,%xmm2 +pxor %xmm2,%xmm13 +movdqa %xmm11,%xmm1 +paddd %xmm9,%xmm1 +movdqa %xmm1,%xmm2 +pslld $13,%xmm1 +pxor %xmm1,%xmm7 +psrld $19,%xmm2 +pxor %xmm2,%xmm7 +movdqa %xmm15,%xmm1 +paddd %xmm13,%xmm1 +movdqa %xmm1,%xmm2 +pslld $18,%xmm1 +pxor %xmm1,%xmm12 +psrld $14,%xmm2 +pxor %xmm2,%xmm12 +movdqa 384(%rsp),%xmm1 +movdqa %xmm12,384(%rsp) +movdqa %xmm9,%xmm2 +paddd %xmm7,%xmm2 +movdqa %xmm2,%xmm12 +pslld $18,%xmm2 +pxor %xmm2,%xmm0 +psrld $14,%xmm12 +pxor %xmm12,%xmm0 +movdqa %xmm5,%xmm2 +paddd %xmm1,%xmm2 +movdqa %xmm2,%xmm12 +pslld $7,%xmm2 +pxor %xmm2,%xmm3 +psrld $25,%xmm12 +pxor %xmm12,%xmm3 +movdqa 400(%rsp),%xmm2 +movdqa %xmm0,400(%rsp) +movdqa %xmm6,%xmm0 +paddd %xmm2,%xmm0 +movdqa %xmm0,%xmm12 +pslld $7,%xmm0 +pxor %xmm0,%xmm4 +psrld $25,%xmm12 +pxor %xmm12,%xmm4 +movdqa %xmm1,%xmm0 +paddd %xmm3,%xmm0 +movdqa %xmm0,%xmm12 +pslld $9,%xmm0 +pxor %xmm0,%xmm10 +psrld $23,%xmm12 +pxor %xmm12,%xmm10 +movdqa %xmm2,%xmm0 +paddd %xmm4,%xmm0 +movdqa %xmm0,%xmm12 +pslld $9,%xmm0 +pxor %xmm0,%xmm8 +psrld $23,%xmm12 +pxor %xmm12,%xmm8 +movdqa %xmm3,%xmm0 +paddd %xmm10,%xmm0 +movdqa %xmm0,%xmm12 +pslld $13,%xmm0 +pxor %xmm0,%xmm5 +psrld $19,%xmm12 +pxor %xmm12,%xmm5 +movdqa %xmm4,%xmm0 +paddd %xmm8,%xmm0 +movdqa %xmm0,%xmm12 +pslld $13,%xmm0 +pxor %xmm0,%xmm6 +psrld $19,%xmm12 +pxor %xmm12,%xmm6 +movdqa %xmm10,%xmm0 +paddd %xmm5,%xmm0 +movdqa %xmm0,%xmm12 +pslld $18,%xmm0 +pxor %xmm0,%xmm1 +psrld $14,%xmm12 +pxor %xmm12,%xmm1 +movdqa 384(%rsp),%xmm0 +movdqa %xmm1,384(%rsp) +movdqa %xmm4,%xmm1 +paddd %xmm0,%xmm1 +movdqa %xmm1,%xmm12 +pslld $7,%xmm1 +pxor %xmm1,%xmm7 +psrld $25,%xmm12 +pxor %xmm12,%xmm7 +movdqa %xmm8,%xmm1 +paddd %xmm6,%xmm1 +movdqa %xmm1,%xmm12 +pslld $18,%xmm1 +pxor %xmm1,%xmm2 +psrld $14,%xmm12 +pxor %xmm12,%xmm2 +movdqa 400(%rsp),%xmm12 +movdqa %xmm2,400(%rsp) +movdqa %xmm14,%xmm1 +paddd %xmm12,%xmm1 +movdqa %xmm1,%xmm2 +pslld $7,%xmm1 +pxor %xmm1,%xmm5 +psrld $25,%xmm2 +pxor %xmm2,%xmm5 +movdqa %xmm0,%xmm1 +paddd %xmm7,%xmm1 +movdqa %xmm1,%xmm2 +pslld $9,%xmm1 +pxor %xmm1,%xmm10 +psrld $23,%xmm2 +pxor %xmm2,%xmm10 +movdqa %xmm12,%xmm1 +paddd %xmm5,%xmm1 +movdqa %xmm1,%xmm2 +pslld $9,%xmm1 +pxor %xmm1,%xmm8 +psrld $23,%xmm2 +pxor %xmm2,%xmm8 +movdqa %xmm7,%xmm1 +paddd %xmm10,%xmm1 +movdqa %xmm1,%xmm2 +pslld $13,%xmm1 +pxor %xmm1,%xmm4 +psrld $19,%xmm2 +pxor %xmm2,%xmm4 +movdqa %xmm5,%xmm1 +paddd %xmm8,%xmm1 +movdqa %xmm1,%xmm2 +pslld $13,%xmm1 +pxor %xmm1,%xmm14 +psrld $19,%xmm2 +pxor %xmm2,%xmm14 +movdqa %xmm10,%xmm1 +paddd %xmm4,%xmm1 +movdqa %xmm1,%xmm2 +pslld $18,%xmm1 +pxor %xmm1,%xmm0 +psrld $14,%xmm2 +pxor %xmm2,%xmm0 +movdqa 384(%rsp),%xmm1 +movdqa %xmm0,384(%rsp) +movdqa %xmm8,%xmm0 +paddd %xmm14,%xmm0 +movdqa %xmm0,%xmm2 +pslld $18,%xmm0 +pxor %xmm0,%xmm12 +psrld $14,%xmm2 +pxor %xmm2,%xmm12 +movdqa %xmm11,%xmm0 +paddd %xmm1,%xmm0 +movdqa %xmm0,%xmm2 +pslld $7,%xmm0 +pxor %xmm0,%xmm6 +psrld $25,%xmm2 +pxor %xmm2,%xmm6 +movdqa 400(%rsp),%xmm2 +movdqa %xmm12,400(%rsp) +movdqa %xmm3,%xmm0 +paddd %xmm2,%xmm0 +movdqa %xmm0,%xmm12 +pslld $7,%xmm0 +pxor %xmm0,%xmm13 +psrld $25,%xmm12 +pxor %xmm12,%xmm13 +movdqa %xmm1,%xmm0 +paddd %xmm6,%xmm0 +movdqa %xmm0,%xmm12 +pslld $9,%xmm0 +pxor %xmm0,%xmm15 +psrld $23,%xmm12 +pxor %xmm12,%xmm15 +movdqa %xmm2,%xmm0 +paddd %xmm13,%xmm0 +movdqa %xmm0,%xmm12 +pslld $9,%xmm0 +pxor %xmm0,%xmm9 +psrld $23,%xmm12 +pxor %xmm12,%xmm9 +movdqa %xmm6,%xmm0 +paddd %xmm15,%xmm0 +movdqa %xmm0,%xmm12 +pslld $13,%xmm0 +pxor %xmm0,%xmm11 +psrld $19,%xmm12 +pxor %xmm12,%xmm11 +movdqa %xmm13,%xmm0 +paddd %xmm9,%xmm0 +movdqa %xmm0,%xmm12 +pslld $13,%xmm0 +pxor %xmm0,%xmm3 +psrld $19,%xmm12 +pxor %xmm12,%xmm3 +movdqa %xmm15,%xmm0 +paddd %xmm11,%xmm0 +movdqa %xmm0,%xmm12 +pslld $18,%xmm0 +pxor %xmm0,%xmm1 +psrld $14,%xmm12 +pxor %xmm12,%xmm1 +movdqa %xmm9,%xmm0 +paddd %xmm3,%xmm0 +movdqa %xmm0,%xmm12 +pslld $18,%xmm0 +pxor %xmm0,%xmm2 +psrld $14,%xmm12 +pxor %xmm12,%xmm2 +movdqa 384(%rsp),%xmm12 +movdqa 400(%rsp),%xmm0 +sub $2,%rdx +ja ._mainloop1 + +paddd 176(%rsp),%xmm12 +paddd 240(%rsp),%xmm7 +paddd 288(%rsp),%xmm10 +paddd 336(%rsp),%xmm4 +movd %xmm12,%rdx +movd %xmm7,%rcx +movd %xmm10,%r8 +movd %xmm4,%r9 +pshufd $0x39,%xmm12,%xmm12 +pshufd $0x39,%xmm7,%xmm7 +pshufd $0x39,%xmm10,%xmm10 +pshufd $0x39,%xmm4,%xmm4 +xorl 0(%rsi),%edx +xorl 4(%rsi),%ecx +xorl 8(%rsi),%r8d +xorl 12(%rsi),%r9d +movl %edx,0(%rdi) +movl %ecx,4(%rdi) +movl %r8d,8(%rdi) +movl %r9d,12(%rdi) +movd %xmm12,%rdx +movd %xmm7,%rcx +movd %xmm10,%r8 +movd %xmm4,%r9 +pshufd $0x39,%xmm12,%xmm12 +pshufd $0x39,%xmm7,%xmm7 +pshufd $0x39,%xmm10,%xmm10 +pshufd $0x39,%xmm4,%xmm4 +xorl 64(%rsi),%edx +xorl 68(%rsi),%ecx +xorl 72(%rsi),%r8d +xorl 76(%rsi),%r9d +movl %edx,64(%rdi) +movl %ecx,68(%rdi) +movl %r8d,72(%rdi) +movl %r9d,76(%rdi) +movd %xmm12,%rdx +movd %xmm7,%rcx +movd %xmm10,%r8 +movd %xmm4,%r9 +pshufd $0x39,%xmm12,%xmm12 +pshufd $0x39,%xmm7,%xmm7 +pshufd $0x39,%xmm10,%xmm10 +pshufd $0x39,%xmm4,%xmm4 +xorl 128(%rsi),%edx +xorl 132(%rsi),%ecx +xorl 136(%rsi),%r8d +xorl 140(%rsi),%r9d +movl %edx,128(%rdi) +movl %ecx,132(%rdi) +movl %r8d,136(%rdi) +movl %r9d,140(%rdi) +movd %xmm12,%rdx +movd %xmm7,%rcx +movd %xmm10,%r8 +movd %xmm4,%r9 +xorl 192(%rsi),%edx +xorl 196(%rsi),%ecx +xorl 200(%rsi),%r8d +xorl 204(%rsi),%r9d +movl %edx,192(%rdi) +movl %ecx,196(%rdi) +movl %r8d,200(%rdi) +movl %r9d,204(%rdi) +paddd 304(%rsp),%xmm14 +paddd 128(%rsp),%xmm0 +paddd 192(%rsp),%xmm5 +paddd 256(%rsp),%xmm8 +movd %xmm14,%rdx +movd %xmm0,%rcx +movd %xmm5,%r8 +movd %xmm8,%r9 +pshufd $0x39,%xmm14,%xmm14 +pshufd $0x39,%xmm0,%xmm0 +pshufd $0x39,%xmm5,%xmm5 +pshufd $0x39,%xmm8,%xmm8 +xorl 16(%rsi),%edx +xorl 20(%rsi),%ecx +xorl 24(%rsi),%r8d +xorl 28(%rsi),%r9d +movl %edx,16(%rdi) +movl %ecx,20(%rdi) +movl %r8d,24(%rdi) +movl %r9d,28(%rdi) +movd %xmm14,%rdx +movd %xmm0,%rcx +movd %xmm5,%r8 +movd %xmm8,%r9 +pshufd $0x39,%xmm14,%xmm14 +pshufd $0x39,%xmm0,%xmm0 +pshufd $0x39,%xmm5,%xmm5 +pshufd $0x39,%xmm8,%xmm8 +xorl 80(%rsi),%edx +xorl 84(%rsi),%ecx +xorl 88(%rsi),%r8d +xorl 92(%rsi),%r9d +movl %edx,80(%rdi) +movl %ecx,84(%rdi) +movl %r8d,88(%rdi) +movl %r9d,92(%rdi) +movd %xmm14,%rdx +movd %xmm0,%rcx +movd %xmm5,%r8 +movd %xmm8,%r9 +pshufd $0x39,%xmm14,%xmm14 +pshufd $0x39,%xmm0,%xmm0 +pshufd $0x39,%xmm5,%xmm5 +pshufd $0x39,%xmm8,%xmm8 +xorl 144(%rsi),%edx +xorl 148(%rsi),%ecx +xorl 152(%rsi),%r8d +xorl 156(%rsi),%r9d +movl %edx,144(%rdi) +movl %ecx,148(%rdi) +movl %r8d,152(%rdi) +movl %r9d,156(%rdi) +movd %xmm14,%rdx +movd %xmm0,%rcx +movd %xmm5,%r8 +movd %xmm8,%r9 +xorl 208(%rsi),%edx +xorl 212(%rsi),%ecx +xorl 216(%rsi),%r8d +xorl 220(%rsi),%r9d +movl %edx,208(%rdi) +movl %ecx,212(%rdi) +movl %r8d,216(%rdi) +movl %r9d,220(%rdi) +paddd 352(%rsp),%xmm15 +paddd 368(%rsp),%xmm11 +paddd 144(%rsp),%xmm1 +paddd 208(%rsp),%xmm6 +movd %xmm15,%rdx +movd %xmm11,%rcx +movd %xmm1,%r8 +movd %xmm6,%r9 +pshufd $0x39,%xmm15,%xmm15 +pshufd $0x39,%xmm11,%xmm11 +pshufd $0x39,%xmm1,%xmm1 +pshufd $0x39,%xmm6,%xmm6 +xorl 32(%rsi),%edx +xorl 36(%rsi),%ecx +xorl 40(%rsi),%r8d +xorl 44(%rsi),%r9d +movl %edx,32(%rdi) +movl %ecx,36(%rdi) +movl %r8d,40(%rdi) +movl %r9d,44(%rdi) +movd %xmm15,%rdx +movd %xmm11,%rcx +movd %xmm1,%r8 +movd %xmm6,%r9 +pshufd $0x39,%xmm15,%xmm15 +pshufd $0x39,%xmm11,%xmm11 +pshufd $0x39,%xmm1,%xmm1 +pshufd $0x39,%xmm6,%xmm6 +xorl 96(%rsi),%edx +xorl 100(%rsi),%ecx +xorl 104(%rsi),%r8d +xorl 108(%rsi),%r9d +movl %edx,96(%rdi) +movl %ecx,100(%rdi) +movl %r8d,104(%rdi) +movl %r9d,108(%rdi) +movd %xmm15,%rdx +movd %xmm11,%rcx +movd %xmm1,%r8 +movd %xmm6,%r9 +pshufd $0x39,%xmm15,%xmm15 +pshufd $0x39,%xmm11,%xmm11 +pshufd $0x39,%xmm1,%xmm1 +pshufd $0x39,%xmm6,%xmm6 +xorl 160(%rsi),%edx +xorl 164(%rsi),%ecx +xorl 168(%rsi),%r8d +xorl 172(%rsi),%r9d +movl %edx,160(%rdi) +movl %ecx,164(%rdi) +movl %r8d,168(%rdi) +movl %r9d,172(%rdi) +movd %xmm15,%rdx +movd %xmm11,%rcx +movd %xmm1,%r8 +movd %xmm6,%r9 +xorl 224(%rsi),%edx +xorl 228(%rsi),%ecx +xorl 232(%rsi),%r8d +xorl 236(%rsi),%r9d +movl %edx,224(%rdi) +movl %ecx,228(%rdi) +movl %r8d,232(%rdi) +movl %r9d,236(%rdi) +paddd 224(%rsp),%xmm13 +paddd 272(%rsp),%xmm9 +paddd 320(%rsp),%xmm3 +paddd 160(%rsp),%xmm2 +movd %xmm13,%rdx +movd %xmm9,%rcx +movd %xmm3,%r8 +movd %xmm2,%r9 +pshufd $0x39,%xmm13,%xmm13 +pshufd $0x39,%xmm9,%xmm9 +pshufd $0x39,%xmm3,%xmm3 +pshufd $0x39,%xmm2,%xmm2 +xorl 48(%rsi),%edx +xorl 52(%rsi),%ecx +xorl 56(%rsi),%r8d +xorl 60(%rsi),%r9d +movl %edx,48(%rdi) +movl %ecx,52(%rdi) +movl %r8d,56(%rdi) +movl %r9d,60(%rdi) +movd %xmm13,%rdx +movd %xmm9,%rcx +movd %xmm3,%r8 +movd %xmm2,%r9 +pshufd $0x39,%xmm13,%xmm13 +pshufd $0x39,%xmm9,%xmm9 +pshufd $0x39,%xmm3,%xmm3 +pshufd $0x39,%xmm2,%xmm2 +xorl 112(%rsi),%edx +xorl 116(%rsi),%ecx +xorl 120(%rsi),%r8d +xorl 124(%rsi),%r9d +movl %edx,112(%rdi) +movl %ecx,116(%rdi) +movl %r8d,120(%rdi) +movl %r9d,124(%rdi) +movd %xmm13,%rdx +movd %xmm9,%rcx +movd %xmm3,%r8 +movd %xmm2,%r9 +pshufd $0x39,%xmm13,%xmm13 +pshufd $0x39,%xmm9,%xmm9 +pshufd $0x39,%xmm3,%xmm3 +pshufd $0x39,%xmm2,%xmm2 +xorl 176(%rsi),%edx +xorl 180(%rsi),%ecx +xorl 184(%rsi),%r8d +xorl 188(%rsi),%r9d +movl %edx,176(%rdi) +movl %ecx,180(%rdi) +movl %r8d,184(%rdi) +movl %r9d,188(%rdi) +movd %xmm13,%rdx +movd %xmm9,%rcx +movd %xmm3,%r8 +movd %xmm2,%r9 +xorl 240(%rsi),%edx +xorl 244(%rsi),%ecx +xorl 248(%rsi),%r8d +xorl 252(%rsi),%r9d +movl %edx,240(%rdi) +movl %ecx,244(%rdi) +movl %r8d,248(%rdi) +movl %r9d,252(%rdi) +movq 480(%rsp),%r9 +sub $256,%r9 +add $256,%rsi +add $256,%rdi +cmp $256,%r9 +jae ._bytesatleast256 + +cmp $0,%r9 +jbe ._done + +._bytesbetween1and255: +cmp $64,%r9 +jae ._nocopy + +mov %rdi,%rdx +leaq 0(%rsp),%rdi +mov %r9,%rcx +rep movsb +leaq 0(%rsp),%rdi +leaq 0(%rsp),%rsi + +._nocopy: +movq %r9,480(%rsp) +movdqa 112(%rsp),%xmm0 +movdqa 64(%rsp),%xmm1 +movdqa 80(%rsp),%xmm2 +movdqa 96(%rsp),%xmm3 +movdqa %xmm1,%xmm4 +mov $20,%rcx + +.p2align 4 +._mainloop2: +paddd %xmm0,%xmm4 +movdqa %xmm0,%xmm5 +movdqa %xmm4,%xmm6 +pslld $7,%xmm4 +psrld $25,%xmm6 +pxor %xmm4,%xmm3 +pxor %xmm6,%xmm3 +paddd %xmm3,%xmm5 +movdqa %xmm3,%xmm4 +movdqa %xmm5,%xmm6 +pslld $9,%xmm5 +psrld $23,%xmm6 +pxor %xmm5,%xmm2 +pshufd $0x93,%xmm3,%xmm3 +pxor %xmm6,%xmm2 +paddd %xmm2,%xmm4 +movdqa %xmm2,%xmm5 +movdqa %xmm4,%xmm6 +pslld $13,%xmm4 +psrld $19,%xmm6 +pxor %xmm4,%xmm1 +pshufd $0x4e,%xmm2,%xmm2 +pxor %xmm6,%xmm1 +paddd %xmm1,%xmm5 +movdqa %xmm3,%xmm4 +movdqa %xmm5,%xmm6 +pslld $18,%xmm5 +psrld $14,%xmm6 +pxor %xmm5,%xmm0 +pshufd $0x39,%xmm1,%xmm1 +pxor %xmm6,%xmm0 +paddd %xmm0,%xmm4 +movdqa %xmm0,%xmm5 +movdqa %xmm4,%xmm6 +pslld $7,%xmm4 +psrld $25,%xmm6 +pxor %xmm4,%xmm1 +pxor %xmm6,%xmm1 +paddd %xmm1,%xmm5 +movdqa %xmm1,%xmm4 +movdqa %xmm5,%xmm6 +pslld $9,%xmm5 +psrld $23,%xmm6 +pxor %xmm5,%xmm2 +pshufd $0x93,%xmm1,%xmm1 +pxor %xmm6,%xmm2 +paddd %xmm2,%xmm4 +movdqa %xmm2,%xmm5 +movdqa %xmm4,%xmm6 +pslld $13,%xmm4 +psrld $19,%xmm6 +pxor %xmm4,%xmm3 +pshufd $0x4e,%xmm2,%xmm2 +pxor %xmm6,%xmm3 +paddd %xmm3,%xmm5 +movdqa %xmm1,%xmm4 +movdqa %xmm5,%xmm6 +pslld $18,%xmm5 +psrld $14,%xmm6 +pxor %xmm5,%xmm0 +pshufd $0x39,%xmm3,%xmm3 +pxor %xmm6,%xmm0 +paddd %xmm0,%xmm4 +movdqa %xmm0,%xmm5 +movdqa %xmm4,%xmm6 +pslld $7,%xmm4 +psrld $25,%xmm6 +pxor %xmm4,%xmm3 +pxor %xmm6,%xmm3 +paddd %xmm3,%xmm5 +movdqa %xmm3,%xmm4 +movdqa %xmm5,%xmm6 +pslld $9,%xmm5 +psrld $23,%xmm6 +pxor %xmm5,%xmm2 +pshufd $0x93,%xmm3,%xmm3 +pxor %xmm6,%xmm2 +paddd %xmm2,%xmm4 +movdqa %xmm2,%xmm5 +movdqa %xmm4,%xmm6 +pslld $13,%xmm4 +psrld $19,%xmm6 +pxor %xmm4,%xmm1 +pshufd $0x4e,%xmm2,%xmm2 +pxor %xmm6,%xmm1 +paddd %xmm1,%xmm5 +movdqa %xmm3,%xmm4 +movdqa %xmm5,%xmm6 +pslld $18,%xmm5 +psrld $14,%xmm6 +pxor %xmm5,%xmm0 +pshufd $0x39,%xmm1,%xmm1 +pxor %xmm6,%xmm0 +paddd %xmm0,%xmm4 +movdqa %xmm0,%xmm5 +movdqa %xmm4,%xmm6 +pslld $7,%xmm4 +psrld $25,%xmm6 +pxor %xmm4,%xmm1 +pxor %xmm6,%xmm1 +paddd %xmm1,%xmm5 +movdqa %xmm1,%xmm4 +movdqa %xmm5,%xmm6 +pslld $9,%xmm5 +psrld $23,%xmm6 +pxor %xmm5,%xmm2 +pshufd $0x93,%xmm1,%xmm1 +pxor %xmm6,%xmm2 +paddd %xmm2,%xmm4 +movdqa %xmm2,%xmm5 +movdqa %xmm4,%xmm6 +pslld $13,%xmm4 +psrld $19,%xmm6 +pxor %xmm4,%xmm3 +pshufd $0x4e,%xmm2,%xmm2 +pxor %xmm6,%xmm3 +sub $4,%rcx +paddd %xmm3,%xmm5 +movdqa %xmm1,%xmm4 +movdqa %xmm5,%xmm6 +pslld $18,%xmm5 +pxor %xmm7,%xmm7 +psrld $14,%xmm6 +pxor %xmm5,%xmm0 +pshufd $0x39,%xmm3,%xmm3 +pxor %xmm6,%xmm0 +ja ._mainloop2 + +paddd 112(%rsp),%xmm0 +paddd 64(%rsp),%xmm1 +paddd 80(%rsp),%xmm2 +paddd 96(%rsp),%xmm3 +movd %xmm0,%rcx +movd %xmm1,%r8 +movd %xmm2,%r9 +movd %xmm3,%rax +pshufd $0x39,%xmm0,%xmm0 +pshufd $0x39,%xmm1,%xmm1 +pshufd $0x39,%xmm2,%xmm2 +pshufd $0x39,%xmm3,%xmm3 +xorl 0(%rsi),%ecx +xorl 48(%rsi),%r8d +xorl 32(%rsi),%r9d +xorl 16(%rsi),%eax +movl %ecx,0(%rdi) +movl %r8d,48(%rdi) +movl %r9d,32(%rdi) +movl %eax,16(%rdi) +movd %xmm0,%rcx +movd %xmm1,%r8 +movd %xmm2,%r9 +movd %xmm3,%rax +pshufd $0x39,%xmm0,%xmm0 +pshufd $0x39,%xmm1,%xmm1 +pshufd $0x39,%xmm2,%xmm2 +pshufd $0x39,%xmm3,%xmm3 +xorl 20(%rsi),%ecx +xorl 4(%rsi),%r8d +xorl 52(%rsi),%r9d +xorl 36(%rsi),%eax +movl %ecx,20(%rdi) +movl %r8d,4(%rdi) +movl %r9d,52(%rdi) +movl %eax,36(%rdi) +movd %xmm0,%rcx +movd %xmm1,%r8 +movd %xmm2,%r9 +movd %xmm3,%rax +pshufd $0x39,%xmm0,%xmm0 +pshufd $0x39,%xmm1,%xmm1 +pshufd $0x39,%xmm2,%xmm2 +pshufd $0x39,%xmm3,%xmm3 +xorl 40(%rsi),%ecx +xorl 24(%rsi),%r8d +xorl 8(%rsi),%r9d +xorl 56(%rsi),%eax +movl %ecx,40(%rdi) +movl %r8d,24(%rdi) +movl %r9d,8(%rdi) +movl %eax,56(%rdi) +movd %xmm0,%rcx +movd %xmm1,%r8 +movd %xmm2,%r9 +movd %xmm3,%rax +xorl 60(%rsi),%ecx +xorl 44(%rsi),%r8d +xorl 28(%rsi),%r9d +xorl 12(%rsi),%eax +movl %ecx,60(%rdi) +movl %r8d,44(%rdi) +movl %r9d,28(%rdi) +movl %eax,12(%rdi) +movq 480(%rsp),%r9 +movq 472(%rsp),%rcx +add $1,%rcx +mov %rcx,%r8 +shr $32,%r8 +movl %ecx,80(%rsp) +movl %r8d,4+96(%rsp) +movq %rcx,472(%rsp) +cmp $64,%r9 +ja ._bytesatleast65 +jae ._bytesatleast64 + +mov %rdi,%rsi +mov %rdx,%rdi +mov %r9,%rcx +rep movsb + +._bytesatleast64: +._done: +movq 416(%rsp),%r11 +movq 424(%rsp),%r12 +movq 432(%rsp),%r13 +movq 440(%rsp),%r14 +movq 448(%rsp),%r15 +movq 456(%rsp),%rbx +movq 464(%rsp),%rbp +add %r11,%rsp +xor %rax,%rax +mov %rsi,%rdx +ret + +._bytesatleast65: +sub $64,%r9 +add $64,%rdi +add $64,%rsi +jmp ._bytesbetween1and255 + +#endif + +#if defined(__linux__) && defined(__ELF__) +.section .note.GNU-stack,"",%progbits +#endif + +.globl consts_S_dummy diff --git a/sodium/sodium_sandy2x.S b/sodium/sodium_sandy2x.S new file mode 100644 index 00000000..49031f08 --- /dev/null +++ b/sodium/sodium_sandy2x.S @@ -0,0 +1,19 @@ + +#ifdef NETCODE_AVX + +#define IN_SANDY2X + +#include "sodium_sandy2x_consts.S" +#include "sodium_sandy2x_fe51_mul.S" +#include "sodium_sandy2x_fe51_nsquare.S" +#include "sodium_sandy2x_fe51_pack.S" +#include "sodium_sandy2x_ladder.S" +#include "sodium_sandy2x_ladder_base.S" + +#if defined(__linux__) && defined(__ELF__) +.section .note.GNU-stack,"",%progbits +#endif + +#endif + +.globl consts_S_dummy diff --git a/sodium/sodium_sandy2x_consts.S b/sodium/sodium_sandy2x_consts.S new file mode 100644 index 00000000..7cd6c3b7 --- /dev/null +++ b/sodium/sodium_sandy2x_consts.S @@ -0,0 +1,27 @@ +#ifdef IN_SANDY2X + +/* + REDMASK51 is from amd64-51/consts.s. +*/ + +#include "sodium_sandy2x_consts_namespace.h" +.data +.p2align 4 +v0_0: .quad 0, 0 +v1_0: .quad 1, 0 +v2_1: .quad 2, 1 +v9_0: .quad 9, 0 +v9_9: .quad 9, 9 +v19_19: .quad 19, 19 +v38_1: .quad 38, 1 +v38_38: .quad 38, 38 +v121666_121666: .quad 121666, 121666 +m25: .quad 33554431, 33554431 +m26: .quad 67108863, 67108863 +subc0: .quad 0x07FFFFDA, 0x03FFFFFE +subc2: .quad 0x07FFFFFE, 0x03FFFFFE +REDMASK51: .quad 0x0007FFFFFFFFFFFF + +#endif + +.globl consts_S_dummy diff --git a/sodium/sodium_sandy2x_consts_namespace.h b/sodium/sodium_sandy2x_consts_namespace.h new file mode 100644 index 00000000..9f81fa61 --- /dev/null +++ b/sodium/sodium_sandy2x_consts_namespace.h @@ -0,0 +1,20 @@ +#ifndef consts_namespace_H +#define consts_namespace_H + +#define v0_0 crypto_scalarmult_curve25519_sandy2x_v0_0 +#define v1_0 crypto_scalarmult_curve25519_sandy2x_v1_0 +#define v2_1 crypto_scalarmult_curve25519_sandy2x_v2_1 +#define v9_0 crypto_scalarmult_curve25519_sandy2x_v9_0 +#define v9_9 crypto_scalarmult_curve25519_sandy2x_v9_9 +#define v19_19 crypto_scalarmult_curve25519_sandy2x_v19_19 +#define v38_1 crypto_scalarmult_curve25519_sandy2x_v38_1 +#define v38_38 crypto_scalarmult_curve25519_sandy2x_v38_38 +#define v121666_121666 crypto_scalarmult_curve25519_sandy2x_v121666_121666 +#define m25 crypto_scalarmult_curve25519_sandy2x_m25 +#define m26 crypto_scalarmult_curve25519_sandy2x_m26 +#define subc0 crypto_scalarmult_curve25519_sandy2x_subc0 +#define subc2 crypto_scalarmult_curve25519_sandy2x_subc2 +#define REDMASK51 crypto_scalarmult_curve25519_sandy2x_REDMASK51 + +#endif /* ifndef consts_namespace_H */ + diff --git a/sodium/sodium_sandy2x_curve25519.c b/sodium/sodium_sandy2x_curve25519.c new file mode 100644 index 00000000..9f83b5ca --- /dev/null +++ b/sodium/sodium_sandy2x_curve25519.c @@ -0,0 +1,118 @@ +/* + This file is adapted from ref10/scalarmult.c: + The code for Mongomery ladder is replace by the ladder assembly function; + Inversion is done in the same way as amd64-51/. + (fe is first converted into fe51 after Mongomery ladder) +*/ + +#include + +#include "sodium_private_common.h" + +#ifdef HAVE_AVX_ASM + +#include "sodium_utils.h" +#include "sodium_sandy2x_curve25519.h" +#include "sodium_scalarmult_curve25519.h" +#include "sodium_sandy2x_fe.h" +#include "sodium_sandy2x_fe51.h" +#include "sodium_sandy2x_ladder.h" +#include "sodium_sandy2x_ladder_base.h" + +#define x1 var[0] +#define x2 var[1] +#define z2 var[2] + +static int +crypto_scalarmult_curve25519_sandy2x(unsigned char *q, const unsigned char *n, + const unsigned char *p) +{ + unsigned char *t = q; + fe var[3]; + fe51 x_51; + fe51 z_51; + unsigned int i; + + for (i = 0; i < 32; i++) { + t[i] = n[i]; + } + t[0] &= 248; + t[31] &= 127; + t[31] |= 64; + + fe_frombytes(x1, p); + + ladder(var, t); + + z_51.v[0] = (z2[1] << 26) + z2[0]; + z_51.v[1] = (z2[3] << 26) + z2[2]; + z_51.v[2] = (z2[5] << 26) + z2[4]; + z_51.v[3] = (z2[7] << 26) + z2[6]; + z_51.v[4] = (z2[9] << 26) + z2[8]; + + x_51.v[0] = (x2[1] << 26) + x2[0]; + x_51.v[1] = (x2[3] << 26) + x2[2]; + x_51.v[2] = (x2[5] << 26) + x2[4]; + x_51.v[3] = (x2[7] << 26) + x2[6]; + x_51.v[4] = (x2[9] << 26) + x2[8]; + + fe51_invert(&z_51, &z_51); + fe51_mul(&x_51, &x_51, &z_51); + fe51_pack(q, &x_51); + + return 0; +} + +#undef x2 +#undef z2 + +#define x2 var[0] +#define z2 var[1] + +static int +crypto_scalarmult_curve25519_sandy2x_base(unsigned char *q, + const unsigned char *n) +{ + unsigned char *t = q; + fe var[3]; + fe51 x_51; + fe51 z_51; + unsigned int i; + + for (i = 0;i < 32; i++) { + t[i] = n[i]; + } + t[0] &= 248; + t[31] &= 127; + t[31] |= 64; + + ladder_base(var, t); + + z_51.v[0] = (z2[1] << 26) + z2[0]; + z_51.v[1] = (z2[3] << 26) + z2[2]; + z_51.v[2] = (z2[5] << 26) + z2[4]; + z_51.v[3] = (z2[7] << 26) + z2[6]; + z_51.v[4] = (z2[9] << 26) + z2[8]; + + x_51.v[0] = (x2[1] << 26) + x2[0]; + x_51.v[1] = (x2[3] << 26) + x2[2]; + x_51.v[2] = (x2[5] << 26) + x2[4]; + x_51.v[3] = (x2[7] << 26) + x2[6]; + x_51.v[4] = (x2[9] << 26) + x2[8]; + + fe51_invert(&z_51, &z_51); + fe51_mul(&x_51, &x_51, &z_51); + fe51_pack(q, &x_51); + + return 0; +} + +struct crypto_scalarmult_curve25519_implementation +crypto_scalarmult_curve25519_sandy2x_implementation = { + SODIUM_C99(.mult = ) crypto_scalarmult_curve25519_sandy2x, + SODIUM_C99(.mult_base = ) crypto_scalarmult_curve25519_sandy2x_base +}; + +#endif + +int curve_25519_sandy2x_link_warning_dummy = 0; diff --git a/sodium/sodium_sandy2x_curve25519.h b/sodium/sodium_sandy2x_curve25519.h new file mode 100644 index 00000000..df7e8a51 --- /dev/null +++ b/sodium/sodium_sandy2x_curve25519.h @@ -0,0 +1,9 @@ +#ifndef curve25519_sandy2x_H +#define curve25519_sandy2x_H + +#include "sodium_crypto_scalarmult_curve25519.h" + +extern struct crypto_scalarmult_curve25519_implementation + crypto_scalarmult_curve25519_sandy2x_implementation; + +#endif diff --git a/sodium/sodium_sandy2x_fe.h b/sodium/sodium_sandy2x_fe.h new file mode 100644 index 00000000..b1115f86 --- /dev/null +++ b/sodium/sodium_sandy2x_fe.h @@ -0,0 +1,26 @@ +/* + This file is adapted from ref10/fe.h: + All the redundant functions are removed. +*/ + +#ifndef fe_H +#define fe_H + +#include +#include + +typedef uint64_t fe[10]; + +/* +fe means field element. +Here the field is \Z/(2^255-19). +An element t, entries t[0]...t[9], represents the integer +t[0]+2^26 t[1]+2^51 t[2]+2^77 t[3]+2^102 t[4]+...+2^230 t[9]. +Bounds on each t[i] vary depending on context. +*/ + +#define fe_frombytes crypto_scalarmult_curve25519_sandy2x_fe_frombytes + +extern void fe_frombytes(fe, const unsigned char *); + +#endif diff --git a/sodium/sodium_sandy2x_fe51.h b/sodium/sodium_sandy2x_fe51.h new file mode 100644 index 00000000..cfb3109e --- /dev/null +++ b/sodium/sodium_sandy2x_fe51.h @@ -0,0 +1,35 @@ +/* + This file is adapted from amd64-51/fe25519.h: + 'fe25519' is renamed as 'fe51'; + All the redundant functions are removed; + New function fe51_nsquare is introduced. +*/ + +#ifndef fe51_H +#define fe51_H + +#ifdef __cplusplus +extern "C" { +#endif + +#include +#include + +#include "sodium_sandy2x_fe51_namespace.h" + +typedef struct +{ + uint64_t v[5]; +} +fe51; + +extern void fe51_pack(unsigned char *, const fe51 *); +extern void fe51_mul(fe51 *, const fe51 *, const fe51 *); +extern void fe51_nsquare(fe51 *, const fe51 *, int); +extern void fe51_invert(fe51 *, const fe51 *); + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/sodium/sodium_sandy2x_fe51_invert.c b/sodium/sodium_sandy2x_fe51_invert.c new file mode 100644 index 00000000..52784d52 --- /dev/null +++ b/sodium/sodium_sandy2x_fe51_invert.c @@ -0,0 +1,61 @@ +/* + This file is adapted from amd64-51/fe25519_invert.c: + Loops of squares are replaced by nsquares for better performance. +*/ + +#include "sodium_sandy2x_fe51.h" +#include "sodium_private_common.h" + +#ifdef HAVE_AVX_ASM + +#define fe51_square(x, y) fe51_nsquare(x, y, 1) + +void +fe51_invert(fe51 *r, const fe51 *x) +{ + fe51 z2; + fe51 z9; + fe51 z11; + fe51 z2_5_0; + fe51 z2_10_0; + fe51 z2_20_0; + fe51 z2_50_0; + fe51 z2_100_0; + fe51 t; + + /* 2 */ fe51_square(&z2,x); + /* 4 */ fe51_square(&t,&z2); + /* 8 */ fe51_square(&t,&t); + /* 9 */ fe51_mul(&z9,&t,x); + /* 11 */ fe51_mul(&z11,&z9,&z2); + /* 22 */ fe51_square(&t,&z11); + /* 2^5 - 2^0 = 31 */ fe51_mul(&z2_5_0,&t,&z9); + + /* 2^10 - 2^5 */ fe51_nsquare(&t,&z2_5_0, 5); + /* 2^10 - 2^0 */ fe51_mul(&z2_10_0,&t,&z2_5_0); + + /* 2^20 - 2^10 */ fe51_nsquare(&t,&z2_10_0, 10); + /* 2^20 - 2^0 */ fe51_mul(&z2_20_0,&t,&z2_10_0); + + /* 2^40 - 2^20 */ fe51_nsquare(&t,&z2_20_0, 20); + /* 2^40 - 2^0 */ fe51_mul(&t,&t,&z2_20_0); + + /* 2^50 - 2^10 */ fe51_nsquare(&t,&t,10); + /* 2^50 - 2^0 */ fe51_mul(&z2_50_0,&t,&z2_10_0); + + /* 2^100 - 2^50 */ fe51_nsquare(&t,&z2_50_0, 50); + /* 2^100 - 2^0 */ fe51_mul(&z2_100_0,&t,&z2_50_0); + + /* 2^200 - 2^100 */ fe51_nsquare(&t,&z2_100_0, 100); + /* 2^200 - 2^0 */ fe51_mul(&t,&t,&z2_100_0); + + /* 2^250 - 2^50 */ fe51_nsquare(&t,&t, 50); + /* 2^250 - 2^0 */ fe51_mul(&t,&t,&z2_50_0); + + /* 2^255 - 2^5 */ fe51_nsquare(&t,&t,5); + /* 2^255 - 21 */ fe51_mul(r,&t,&z11); +} + +#endif + +int fe51_invert_link_warning_dummy = 0; diff --git a/sodium/sodium_sandy2x_fe51_mul.S b/sodium/sodium_sandy2x_fe51_mul.S new file mode 100644 index 00000000..c2a5739e --- /dev/null +++ b/sodium/sodium_sandy2x_fe51_mul.S @@ -0,0 +1,199 @@ +#ifdef IN_SANDY2X + +/* + This file is basically amd64-51/fe25519_mul.s. +*/ +#include "sodium_sandy2x_fe51_namespace.h" +#include "sodium_sandy2x_consts_namespace.h" +.text +.p2align 5 +#ifdef ASM_HIDE_SYMBOL +ASM_HIDE_SYMBOL fe51_mul +ASM_HIDE_SYMBOL _fe51_mul +#endif +.globl fe51_mul +.globl _fe51_mul +#ifdef __ELF__ +.type fe51_mul, @function +.type _fe51_mul, @function +#endif +fe51_mul: +_fe51_mul: +mov %rsp,%r11 +and $31,%r11 +add $96,%r11 +sub %r11,%rsp +movq %r11,0(%rsp) +movq %r12,8(%rsp) +movq %r13,16(%rsp) +movq %r14,24(%rsp) +movq %r15,32(%rsp) +movq %rbx,40(%rsp) +movq %rbp,48(%rsp) +movq %rdi,56(%rsp) +mov %rdx,%rcx +movq 24(%rsi),%rdx +imulq $19,%rdx,%rax +movq %rax,64(%rsp) +mulq 16(%rcx) +mov %rax,%r8 +mov %rdx,%r9 +movq 32(%rsi),%rdx +imulq $19,%rdx,%rax +movq %rax,72(%rsp) +mulq 8(%rcx) +add %rax,%r8 +adc %rdx,%r9 +movq 0(%rsi),%rax +mulq 0(%rcx) +add %rax,%r8 +adc %rdx,%r9 +movq 0(%rsi),%rax +mulq 8(%rcx) +mov %rax,%r10 +mov %rdx,%r11 +movq 0(%rsi),%rax +mulq 16(%rcx) +mov %rax,%r12 +mov %rdx,%r13 +movq 0(%rsi),%rax +mulq 24(%rcx) +mov %rax,%r14 +mov %rdx,%r15 +movq 0(%rsi),%rax +mulq 32(%rcx) +mov %rax,%rbx +mov %rdx,%rbp +movq 8(%rsi),%rax +mulq 0(%rcx) +add %rax,%r10 +adc %rdx,%r11 +movq 8(%rsi),%rax +mulq 8(%rcx) +add %rax,%r12 +adc %rdx,%r13 +movq 8(%rsi),%rax +mulq 16(%rcx) +add %rax,%r14 +adc %rdx,%r15 +movq 8(%rsi),%rax +mulq 24(%rcx) +add %rax,%rbx +adc %rdx,%rbp +movq 8(%rsi),%rdx +imulq $19,%rdx,%rax +mulq 32(%rcx) +add %rax,%r8 +adc %rdx,%r9 +movq 16(%rsi),%rax +mulq 0(%rcx) +add %rax,%r12 +adc %rdx,%r13 +movq 16(%rsi),%rax +mulq 8(%rcx) +add %rax,%r14 +adc %rdx,%r15 +movq 16(%rsi),%rax +mulq 16(%rcx) +add %rax,%rbx +adc %rdx,%rbp +movq 16(%rsi),%rdx +imulq $19,%rdx,%rax +mulq 24(%rcx) +add %rax,%r8 +adc %rdx,%r9 +movq 16(%rsi),%rdx +imulq $19,%rdx,%rax +mulq 32(%rcx) +add %rax,%r10 +adc %rdx,%r11 +movq 24(%rsi),%rax +mulq 0(%rcx) +add %rax,%r14 +adc %rdx,%r15 +movq 24(%rsi),%rax +mulq 8(%rcx) +add %rax,%rbx +adc %rdx,%rbp +movq 64(%rsp),%rax +mulq 24(%rcx) +add %rax,%r10 +adc %rdx,%r11 +movq 64(%rsp),%rax +mulq 32(%rcx) +add %rax,%r12 +adc %rdx,%r13 +movq 32(%rsi),%rax +mulq 0(%rcx) +add %rax,%rbx +adc %rdx,%rbp +movq 72(%rsp),%rax +mulq 16(%rcx) +add %rax,%r10 +adc %rdx,%r11 +movq 72(%rsp),%rax +mulq 24(%rcx) +add %rax,%r12 +adc %rdx,%r13 +movq 72(%rsp),%rax +mulq 32(%rcx) +add %rax,%r14 +adc %rdx,%r15 +movq REDMASK51(%rip),%rsi +shld $13,%r8,%r9 +and %rsi,%r8 +shld $13,%r10,%r11 +and %rsi,%r10 +add %r9,%r10 +shld $13,%r12,%r13 +and %rsi,%r12 +add %r11,%r12 +shld $13,%r14,%r15 +and %rsi,%r14 +add %r13,%r14 +shld $13,%rbx,%rbp +and %rsi,%rbx +add %r15,%rbx +imulq $19,%rbp,%rdx +add %rdx,%r8 +mov %r8,%rdx +shr $51,%rdx +add %r10,%rdx +mov %rdx,%rcx +shr $51,%rdx +and %rsi,%r8 +add %r12,%rdx +mov %rdx,%r9 +shr $51,%rdx +and %rsi,%rcx +add %r14,%rdx +mov %rdx,%rax +shr $51,%rdx +and %rsi,%r9 +add %rbx,%rdx +mov %rdx,%r10 +shr $51,%rdx +and %rsi,%rax +imulq $19,%rdx,%rdx +add %rdx,%r8 +and %rsi,%r10 +movq %r8,0(%rdi) +movq %rcx,8(%rdi) +movq %r9,16(%rdi) +movq %rax,24(%rdi) +movq %r10,32(%rdi) +movq 0(%rsp),%r11 +movq 8(%rsp),%r12 +movq 16(%rsp),%r13 +movq 24(%rsp),%r14 +movq 32(%rsp),%r15 +movq 40(%rsp),%rbx +movq 48(%rsp),%rbp +add %r11,%rsp +mov %rdi,%rax +mov %rsi,%rdx +ret + +#endif + +.globl fe51_mul_S_dummy diff --git a/sodium/sodium_sandy2x_fe51_namespace.h b/sodium/sodium_sandy2x_fe51_namespace.h new file mode 100644 index 00000000..057f242c --- /dev/null +++ b/sodium/sodium_sandy2x_fe51_namespace.h @@ -0,0 +1,16 @@ +#ifndef fe51_namespace_H +#define fe51_namespace_H + +#define fe51 crypto_scalarmult_curve25519_sandy2x_fe51 +#define _fe51 _crypto_scalarmult_curve25519_sandy2x_fe51 +#define fe51_pack crypto_scalarmult_curve25519_sandy2x_fe51_pack +#define _fe51_pack _crypto_scalarmult_curve25519_sandy2x_fe51_pack +#define fe51_mul crypto_scalarmult_curve25519_sandy2x_fe51_mul +#define _fe51_mul _crypto_scalarmult_curve25519_sandy2x_fe51_mul +#define fe51_nsquare crypto_scalarmult_curve25519_sandy2x_fe51_nsquare +#define _fe51_nsquare _crypto_scalarmult_curve25519_sandy2x_fe51_nsquare + +#define fe51_invert crypto_scalarmult_curve25519_sandy2x_fe51_invert + +#endif /* ifndef fe51_namespace_H */ + diff --git a/sodium/sodium_sandy2x_fe51_nsquare.S b/sodium/sodium_sandy2x_fe51_nsquare.S new file mode 100644 index 00000000..7751a75c --- /dev/null +++ b/sodium/sodium_sandy2x_fe51_nsquare.S @@ -0,0 +1,174 @@ +#ifdef IN_SANDY2X + +/* + This file is adapted from amd64-51/fe25519_square.s: + Adding loop to perform n squares. +*/ +#include "sodium_sandy2x_fe51_namespace.h" +#include "sodium_sandy2x_consts_namespace.h" +.p2align 5 + +#ifdef ASM_HIDE_SYMBOL +ASM_HIDE_SYMBOL fe51_nsquare +ASM_HIDE_SYMBOL _fe51_nsquare +#endif +.globl fe51_nsquare +.globl _fe51_nsquare +#ifdef __ELF__ +.type fe51_nsquare, @function +.type _fe51_nsquare, @function +#endif +fe51_nsquare: +_fe51_nsquare: + +mov %rsp,%r11 +and $31,%r11 +add $64,%r11 +sub %r11,%rsp +movq %r11,0(%rsp) +movq %r12,8(%rsp) +movq %r13,16(%rsp) +movq %r14,24(%rsp) +movq %r15,32(%rsp) +movq %rbx,40(%rsp) +movq %rbp,48(%rsp) +movq 0(%rsi),%rcx +movq 8(%rsi),%r8 +movq 16(%rsi),%r9 +movq 24(%rsi),%rax +movq 32(%rsi),%rsi +movq %r9,16(%rdi) +movq %rax,24(%rdi) +movq %rsi,32(%rdi) +mov %rdx,%rsi + +.p2align 4 +._loop: +sub $1,%rsi +mov %rcx,%rax +mul %rcx +add %rcx,%rcx +mov %rax,%r9 +mov %rdx,%r10 +mov %rcx,%rax +mul %r8 +mov %rax,%r11 +mov %rdx,%r12 +mov %rcx,%rax +mulq 16(%rdi) +mov %rax,%r13 +mov %rdx,%r14 +mov %rcx,%rax +mulq 24(%rdi) +mov %rax,%r15 +mov %rdx,%rbx +mov %rcx,%rax +mulq 32(%rdi) +mov %rax,%rcx +mov %rdx,%rbp +mov %r8,%rax +mul %r8 +add %r8,%r8 +add %rax,%r13 +adc %rdx,%r14 +mov %r8,%rax +mulq 16(%rdi) +add %rax,%r15 +adc %rdx,%rbx +mov %r8,%rax +imulq $19, %r8,%r8 +mulq 24(%rdi) +add %rax,%rcx +adc %rdx,%rbp +mov %r8,%rax +mulq 32(%rdi) +add %rax,%r9 +adc %rdx,%r10 +movq 16(%rdi),%rax +mulq 16(%rdi) +add %rax,%rcx +adc %rdx,%rbp +shld $13,%rcx,%rbp +movq 16(%rdi),%rax +imulq $38, %rax,%rax +mulq 24(%rdi) +add %rax,%r9 +adc %rdx,%r10 +shld $13,%r9,%r10 +movq 16(%rdi),%rax +imulq $38, %rax,%rax +mulq 32(%rdi) +add %rax,%r11 +adc %rdx,%r12 +movq 24(%rdi),%rax +imulq $19, %rax,%rax +mulq 24(%rdi) +add %rax,%r11 +adc %rdx,%r12 +shld $13,%r11,%r12 +movq 24(%rdi),%rax +imulq $38, %rax,%rax +mulq 32(%rdi) +add %rax,%r13 +adc %rdx,%r14 +shld $13,%r13,%r14 +movq 32(%rdi),%rax +imulq $19, %rax,%rax +mulq 32(%rdi) +add %rax,%r15 +adc %rdx,%rbx +shld $13,%r15,%rbx +movq REDMASK51(%rip),%rdx +and %rdx,%rcx +add %rbx,%rcx +and %rdx,%r9 +and %rdx,%r11 +add %r10,%r11 +and %rdx,%r13 +add %r12,%r13 +and %rdx,%r15 +add %r14,%r15 +imulq $19, %rbp,%rbp +lea (%r9,%rbp),%r9 +mov %r9,%rax +shr $51,%r9 +add %r11,%r9 +and %rdx,%rax +mov %r9,%r8 +shr $51,%r9 +add %r13,%r9 +and %rdx,%r8 +mov %r9,%r10 +shr $51,%r9 +add %r15,%r9 +and %rdx,%r10 +movq %r10,16(%rdi) +mov %r9,%r10 +shr $51,%r9 +add %rcx,%r9 +and %rdx,%r10 +movq %r10,24(%rdi) +mov %r9,%r10 +shr $51,%r9 +imulq $19, %r9,%r9 +lea (%rax,%r9),%rcx +and %rdx,%r10 +movq %r10,32(%rdi) +cmp $0,%rsi +jne ._loop + +movq %rcx,0(%rdi) +movq %r8,8(%rdi) +movq 0(%rsp),%r11 +movq 8(%rsp),%r12 +movq 16(%rsp),%r13 +movq 24(%rsp),%r14 +movq 32(%rsp),%r15 +movq 40(%rsp),%rbx +movq 48(%rsp),%rbp +add %r11,%rsp +ret + +#endif + +.globl fe51_nsquare_S_dummy diff --git a/sodium/sodium_sandy2x_fe51_pack.S b/sodium/sodium_sandy2x_fe51_pack.S new file mode 100644 index 00000000..89cff563 --- /dev/null +++ b/sodium/sodium_sandy2x_fe51_pack.S @@ -0,0 +1,228 @@ +#ifdef IN_SANDY2X + +/* + This file is the result of merging + amd64-51/fe25519_pack.c and amd64-51/fe25519_freeze.s. +*/ +#include "sodium_sandy2x_fe51_namespace.h" +#include "sodium_sandy2x_consts_namespace.h" +.p2align 5 + +#ifdef ASM_HIDE_SYMBOL +ASM_HIDE_SYMBOL fe51_pack +ASM_HIDE_SYMBOL _fe51_pack +#endif +.globl fe51_pack +.globl _fe51_pack +#ifdef __ELF__ +.type fe51_pack, @function +.type _fe51_pack, @function +#endif +fe51_pack: +_fe51_pack: + +mov %rsp,%r11 +and $31,%r11 +add $32,%r11 +sub %r11,%rsp +movq %r11,0(%rsp) +movq %r12,8(%rsp) +movq 0(%rsi),%rdx +movq 8(%rsi),%rcx +movq 16(%rsi),%r8 +movq 24(%rsi),%r9 +movq 32(%rsi),%rsi +movq REDMASK51(%rip),%rax +lea -18(%rax),%r10 +mov $3,%r11 + +.p2align 4 +._reduceloop: +mov %rdx,%r12 +shr $51,%r12 +and %rax,%rdx +add %r12,%rcx +mov %rcx,%r12 +shr $51,%r12 +and %rax,%rcx +add %r12,%r8 +mov %r8,%r12 +shr $51,%r12 +and %rax,%r8 +add %r12,%r9 +mov %r9,%r12 +shr $51,%r12 +and %rax,%r9 +add %r12,%rsi +mov %rsi,%r12 +shr $51,%r12 +and %rax,%rsi +imulq $19, %r12,%r12 +add %r12,%rdx +sub $1,%r11 +ja ._reduceloop + +mov $1,%r12 +cmp %r10,%rdx +cmovl %r11,%r12 +cmp %rax,%rcx +cmovne %r11,%r12 +cmp %rax,%r8 +cmovne %r11,%r12 +cmp %rax,%r9 +cmovne %r11,%r12 +cmp %rax,%rsi +cmovne %r11,%r12 +neg %r12 +and %r12,%rax +and %r12,%r10 +sub %r10,%rdx +sub %rax,%rcx +sub %rax,%r8 +sub %rax,%r9 +sub %rax,%rsi +mov %rdx,%rax +and $0xFF,%eax +movb %al,0(%rdi) +mov %rdx,%rax +shr $8,%rax +and $0xFF,%eax +movb %al,1(%rdi) +mov %rdx,%rax +shr $16,%rax +and $0xFF,%eax +movb %al,2(%rdi) +mov %rdx,%rax +shr $24,%rax +and $0xFF,%eax +movb %al,3(%rdi) +mov %rdx,%rax +shr $32,%rax +and $0xFF,%eax +movb %al,4(%rdi) +mov %rdx,%rax +shr $40,%rax +and $0xFF,%eax +movb %al,5(%rdi) +mov %rdx,%rdx +shr $48,%rdx +mov %rcx,%rax +shl $3,%rax +and $0xF8,%eax +xor %rdx,%rax +movb %al,6(%rdi) +mov %rcx,%rdx +shr $5,%rdx +and $0xFF,%edx +movb %dl,7(%rdi) +mov %rcx,%rdx +shr $13,%rdx +and $0xFF,%edx +movb %dl,8(%rdi) +mov %rcx,%rdx +shr $21,%rdx +and $0xFF,%edx +movb %dl,9(%rdi) +mov %rcx,%rdx +shr $29,%rdx +and $0xFF,%edx +movb %dl,10(%rdi) +mov %rcx,%rdx +shr $37,%rdx +and $0xFF,%edx +movb %dl,11(%rdi) +mov %rcx,%rdx +shr $45,%rdx +mov %r8,%rcx +shl $6,%rcx +and $0xC0,%ecx +xor %rdx,%rcx +movb %cl,12(%rdi) +mov %r8,%rdx +shr $2,%rdx +and $0xFF,%edx +movb %dl,13(%rdi) +mov %r8,%rdx +shr $10,%rdx +and $0xFF,%edx +movb %dl,14(%rdi) +mov %r8,%rdx +shr $18,%rdx +and $0xFF,%edx +movb %dl,15(%rdi) +mov %r8,%rdx +shr $26,%rdx +and $0xFF,%edx +movb %dl,16(%rdi) +mov %r8,%rdx +shr $34,%rdx +and $0xFF,%edx +movb %dl,17(%rdi) +mov %r8,%rdx +shr $42,%rdx +movb %dl,18(%rdi) +mov %r8,%rdx +shr $50,%rdx +mov %r9,%rcx +shl $1,%rcx +and $0xFE,%ecx +xor %rdx,%rcx +movb %cl,19(%rdi) +mov %r9,%rdx +shr $7,%rdx +and $0xFF,%edx +movb %dl,20(%rdi) +mov %r9,%rdx +shr $15,%rdx +and $0xFF,%edx +movb %dl,21(%rdi) +mov %r9,%rdx +shr $23,%rdx +and $0xFF,%edx +movb %dl,22(%rdi) +mov %r9,%rdx +shr $31,%rdx +and $0xFF,%edx +movb %dl,23(%rdi) +mov %r9,%rdx +shr $39,%rdx +and $0xFF,%edx +movb %dl,24(%rdi) +mov %r9,%rdx +shr $47,%rdx +mov %rsi,%rcx +shl $4,%rcx +and $0xF0,%ecx +xor %rdx,%rcx +movb %cl,25(%rdi) +mov %rsi,%rdx +shr $4,%rdx +and $0xFF,%edx +movb %dl,26(%rdi) +mov %rsi,%rdx +shr $12,%rdx +and $0xFF,%edx +movb %dl,27(%rdi) +mov %rsi,%rdx +shr $20,%rdx +and $0xFF,%edx +movb %dl,28(%rdi) +mov %rsi,%rdx +shr $28,%rdx +and $0xFF,%edx +movb %dl,29(%rdi) +mov %rsi,%rdx +shr $36,%rdx +and $0xFF,%edx +movb %dl,30(%rdi) +mov %rsi,%rsi +shr $44,%rsi +movb %sil,31(%rdi) +movq 0(%rsp),%r11 +movq 8(%rsp),%r12 +add %r11,%rsp +ret + +#endif + +.globl fe51_pack_S_dummy diff --git a/sodium/sodium_sandy2x_fe_frombytes.c b/sodium/sodium_sandy2x_fe_frombytes.c new file mode 100644 index 00000000..7973b8a9 --- /dev/null +++ b/sodium/sodium_sandy2x_fe_frombytes.c @@ -0,0 +1,81 @@ +/* + This file is basically ref10/fe_frombytes.h. +*/ + +#include "sodium_sandy2x_fe.h" +#include "sodium_private_common.h" + +#ifdef HAVE_AVX_ASM + +static uint64_t +load_3(const unsigned char *in) +{ + uint64_t result; + result = (uint64_t) in[0]; + result |= ((uint64_t) in[1]) << 8; + result |= ((uint64_t) in[2]) << 16; + return result; +} + +static uint64_t +load_4(const unsigned char *in) +{ + uint64_t result; + result = (uint64_t) in[0]; + result |= ((uint64_t) in[1]) << 8; + result |= ((uint64_t) in[2]) << 16; + result |= ((uint64_t) in[3]) << 24; + return result; +} + +void +fe_frombytes(fe h, const unsigned char *s) +{ + uint64_t h0 = load_4(s); + uint64_t h1 = load_3(s + 4) << 6; + uint64_t h2 = load_3(s + 7) << 5; + uint64_t h3 = load_3(s + 10) << 3; + uint64_t h4 = load_3(s + 13) << 2; + uint64_t h5 = load_4(s + 16); + uint64_t h6 = load_3(s + 20) << 7; + uint64_t h7 = load_3(s + 23) << 5; + uint64_t h8 = load_3(s + 26) << 4; + uint64_t h9 = (load_3(s + 29) & 8388607) << 2; + uint64_t carry0; + uint64_t carry1; + uint64_t carry2; + uint64_t carry3; + uint64_t carry4; + uint64_t carry5; + uint64_t carry6; + uint64_t carry7; + uint64_t carry8; + uint64_t carry9; + + carry9 = h9 >> 25; h0 += carry9 * 19; h9 &= 0x1FFFFFF; + carry1 = h1 >> 25; h2 += carry1; h1 &= 0x1FFFFFF; + carry3 = h3 >> 25; h4 += carry3; h3 &= 0x1FFFFFF; + carry5 = h5 >> 25; h6 += carry5; h5 &= 0x1FFFFFF; + carry7 = h7 >> 25; h8 += carry7; h7 &= 0x1FFFFFF; + + carry0 = h0 >> 26; h1 += carry0; h0 &= 0x3FFFFFF; + carry2 = h2 >> 26; h3 += carry2; h2 &= 0x3FFFFFF; + carry4 = h4 >> 26; h5 += carry4; h4 &= 0x3FFFFFF; + carry6 = h6 >> 26; h7 += carry6; h6 &= 0x3FFFFFF; + carry8 = h8 >> 26; h9 += carry8; h8 &= 0x3FFFFFF; + + h[0] = h0; + h[1] = h1; + h[2] = h2; + h[3] = h3; + h[4] = h4; + h[5] = h5; + h[6] = h6; + h[7] = h7; + h[8] = h8; + h[9] = h9; +} + +#endif + +int fe_frombytes_sandy2x_link_warning_dummy = 0; diff --git a/sodium/sodium_sandy2x_ladder.S b/sodium/sodium_sandy2x_ladder.S new file mode 100644 index 00000000..c93a85f8 --- /dev/null +++ b/sodium/sodium_sandy2x_ladder.S @@ -0,0 +1,1442 @@ +#ifdef IN_SANDY2X + +#include "sodium_sandy2x_ladder_namespace.h" +#include "sodium_sandy2x_consts_namespace.h" +.p2align 5 + +#ifdef ASM_HIDE_SYMBOL +ASM_HIDE_SYMBOL ladder +ASM_HIDE_SYMBOL _ladder +#endif +.globl ladder +.globl _ladder +#ifdef __ELF__ +.type ladder, @function +.type _ladder, @function +#endif +ladder: +_ladder: + +mov %rsp,%r11 +and $31,%r11 +add $1856,%r11 +sub %r11,%rsp +movq %r11,1824(%rsp) +movq %r12,1832(%rsp) +movq %r13,1840(%rsp) +movq %r14,1848(%rsp) +vmovdqa v0_0(%rip),%xmm0 +vmovdqa v1_0(%rip),%xmm1 +vmovdqu 0(%rdi),%xmm2 +vmovdqa %xmm2,0(%rsp) +vmovdqu 16(%rdi),%xmm2 +vmovdqa %xmm2,16(%rsp) +vmovdqu 32(%rdi),%xmm2 +vmovdqa %xmm2,32(%rsp) +vmovdqu 48(%rdi),%xmm2 +vmovdqa %xmm2,48(%rsp) +vmovdqu 64(%rdi),%xmm2 +vmovdqa %xmm2,64(%rsp) +vmovdqa %xmm1,80(%rsp) +vmovdqa %xmm0,96(%rsp) +vmovdqa %xmm0,112(%rsp) +vmovdqa %xmm0,128(%rsp) +vmovdqa %xmm0,144(%rsp) +vmovdqa %xmm1,%xmm0 +vpxor %xmm1,%xmm1,%xmm1 +vpxor %xmm2,%xmm2,%xmm2 +vpxor %xmm3,%xmm3,%xmm3 +vpxor %xmm4,%xmm4,%xmm4 +vpxor %xmm5,%xmm5,%xmm5 +vpxor %xmm6,%xmm6,%xmm6 +vpxor %xmm7,%xmm7,%xmm7 +vpxor %xmm8,%xmm8,%xmm8 +vpxor %xmm9,%xmm9,%xmm9 +vmovdqu 0(%rdi),%xmm10 +vmovdqa %xmm10,160(%rsp) +vmovdqu 16(%rdi),%xmm10 +vmovdqa %xmm10,176(%rsp) +vpmuludq v19_19(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,192(%rsp) +vmovdqu 32(%rdi),%xmm10 +vmovdqa %xmm10,208(%rsp) +vpmuludq v19_19(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,224(%rsp) +vmovdqu 48(%rdi),%xmm10 +vmovdqa %xmm10,240(%rsp) +vpmuludq v19_19(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,256(%rsp) +vmovdqu 64(%rdi),%xmm10 +vmovdqa %xmm10,272(%rsp) +vpmuludq v19_19(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,288(%rsp) +vmovdqu 8(%rdi),%xmm10 +vpmuludq v2_1(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,304(%rsp) +vpmuludq v19_19(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,320(%rsp) +vmovdqu 24(%rdi),%xmm10 +vpmuludq v2_1(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,336(%rsp) +vpmuludq v19_19(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,352(%rsp) +vmovdqu 40(%rdi),%xmm10 +vpmuludq v2_1(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,368(%rsp) +vpmuludq v19_19(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,384(%rsp) +vmovdqu 56(%rdi),%xmm10 +vpmuludq v2_1(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,400(%rsp) +vpmuludq v19_19(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,416(%rsp) +vmovdqu 0(%rdi),%xmm10 +vmovdqu 64(%rdi),%xmm11 +vblendps $12, %xmm11, %xmm10, %xmm10 +vpshufd $2,%xmm10,%xmm10 +vpmuludq v38_1(%rip),%xmm10,%xmm10 +vmovdqa %xmm10,432(%rsp) +movq 0(%rsi),%rdx +movq 8(%rsi),%rcx +movq 16(%rsi),%r8 +movq 24(%rsi),%r9 +shrd $1,%rcx,%rdx +shrd $1,%r8,%rcx +shrd $1,%r9,%r8 +shr $1,%r9 +xorq 0(%rsi),%rdx +xorq 8(%rsi),%rcx +xorq 16(%rsi),%r8 +xorq 24(%rsi),%r9 +leaq 800(%rsp),%rsi +mov $64,%rax + +.p2align 4 +._ladder_small_loop: +mov %rdx,%r10 +mov %rcx,%r11 +mov %r8,%r12 +mov %r9,%r13 +shr $1,%rdx +shr $1,%rcx +shr $1,%r8 +shr $1,%r9 +and $1,%r10d +and $1,%r11d +and $1,%r12d +and $1,%r13d +neg %r10 +neg %r11 +neg %r12 +neg %r13 +movl %r10d,0(%rsi) +movl %r11d,256(%rsi) +movl %r12d,512(%rsi) +movl %r13d,768(%rsi) +add $4,%rsi +sub $1,%rax +jne ._ladder_small_loop +mov $255,%rdx +add $760,%rsi + +.p2align 4 +._ladder_loop: +sub $1,%rdx +vbroadcastss 0(%rsi),%xmm10 +sub $4,%rsi +vmovdqa 0(%rsp),%xmm11 +vmovdqa 80(%rsp),%xmm12 +vpxor %xmm11,%xmm0,%xmm13 +vpand %xmm10,%xmm13,%xmm13 +vpxor %xmm13,%xmm0,%xmm0 +vpxor %xmm13,%xmm11,%xmm11 +vpxor %xmm12,%xmm1,%xmm13 +vpand %xmm10,%xmm13,%xmm13 +vpxor %xmm13,%xmm1,%xmm1 +vpxor %xmm13,%xmm12,%xmm12 +vmovdqa 16(%rsp),%xmm13 +vmovdqa 96(%rsp),%xmm14 +vpxor %xmm13,%xmm2,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm2,%xmm2 +vpxor %xmm15,%xmm13,%xmm13 +vpxor %xmm14,%xmm3,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm3,%xmm3 +vpxor %xmm15,%xmm14,%xmm14 +vmovdqa %xmm13,0(%rsp) +vmovdqa %xmm14,16(%rsp) +vmovdqa 32(%rsp),%xmm13 +vmovdqa 112(%rsp),%xmm14 +vpxor %xmm13,%xmm4,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm4,%xmm4 +vpxor %xmm15,%xmm13,%xmm13 +vpxor %xmm14,%xmm5,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm5,%xmm5 +vpxor %xmm15,%xmm14,%xmm14 +vmovdqa %xmm13,32(%rsp) +vmovdqa %xmm14,80(%rsp) +vmovdqa 48(%rsp),%xmm13 +vmovdqa 128(%rsp),%xmm14 +vpxor %xmm13,%xmm6,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm6,%xmm6 +vpxor %xmm15,%xmm13,%xmm13 +vpxor %xmm14,%xmm7,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm7,%xmm7 +vpxor %xmm15,%xmm14,%xmm14 +vmovdqa %xmm13,48(%rsp) +vmovdqa %xmm14,96(%rsp) +vmovdqa 64(%rsp),%xmm13 +vmovdqa 144(%rsp),%xmm14 +vpxor %xmm13,%xmm8,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm8,%xmm8 +vpxor %xmm15,%xmm13,%xmm13 +vpxor %xmm14,%xmm9,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm9,%xmm9 +vpxor %xmm15,%xmm14,%xmm14 +vmovdqa %xmm13,64(%rsp) +vmovdqa %xmm14,112(%rsp) +vpaddq subc0(%rip),%xmm11,%xmm10 +vpsubq %xmm12,%xmm10,%xmm10 +vpaddq %xmm12,%xmm11,%xmm11 +vpunpckhqdq %xmm10,%xmm11,%xmm12 +vpunpcklqdq %xmm10,%xmm11,%xmm10 +vpaddq %xmm1,%xmm0,%xmm11 +vpaddq subc0(%rip),%xmm0,%xmm0 +vpsubq %xmm1,%xmm0,%xmm0 +vpunpckhqdq %xmm11,%xmm0,%xmm1 +vpunpcklqdq %xmm11,%xmm0,%xmm0 +vpmuludq %xmm0,%xmm10,%xmm11 +vpmuludq %xmm1,%xmm10,%xmm13 +vmovdqa %xmm1,128(%rsp) +vpaddq %xmm1,%xmm1,%xmm1 +vpmuludq %xmm0,%xmm12,%xmm14 +vmovdqa %xmm0,144(%rsp) +vpaddq %xmm14,%xmm13,%xmm13 +vpmuludq %xmm1,%xmm12,%xmm0 +vmovdqa %xmm1,448(%rsp) +vpaddq %xmm3,%xmm2,%xmm1 +vpaddq subc2(%rip),%xmm2,%xmm2 +vpsubq %xmm3,%xmm2,%xmm2 +vpunpckhqdq %xmm1,%xmm2,%xmm3 +vpunpcklqdq %xmm1,%xmm2,%xmm1 +vpmuludq %xmm1,%xmm10,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vpmuludq %xmm3,%xmm10,%xmm2 +vmovdqa %xmm3,464(%rsp) +vpaddq %xmm3,%xmm3,%xmm3 +vpmuludq %xmm1,%xmm12,%xmm14 +vmovdqa %xmm1,480(%rsp) +vpaddq %xmm14,%xmm2,%xmm2 +vpmuludq %xmm3,%xmm12,%xmm1 +vmovdqa %xmm3,496(%rsp) +vpaddq %xmm5,%xmm4,%xmm3 +vpaddq subc2(%rip),%xmm4,%xmm4 +vpsubq %xmm5,%xmm4,%xmm4 +vpunpckhqdq %xmm3,%xmm4,%xmm5 +vpunpcklqdq %xmm3,%xmm4,%xmm3 +vpmuludq %xmm3,%xmm10,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vpmuludq %xmm5,%xmm10,%xmm4 +vmovdqa %xmm5,512(%rsp) +vpaddq %xmm5,%xmm5,%xmm5 +vpmuludq %xmm3,%xmm12,%xmm14 +vmovdqa %xmm3,528(%rsp) +vpaddq %xmm14,%xmm4,%xmm4 +vpaddq %xmm7,%xmm6,%xmm3 +vpaddq subc2(%rip),%xmm6,%xmm6 +vpsubq %xmm7,%xmm6,%xmm6 +vpunpckhqdq %xmm3,%xmm6,%xmm7 +vpunpcklqdq %xmm3,%xmm6,%xmm3 +vpmuludq %xmm3,%xmm10,%xmm6 +vpmuludq %xmm5,%xmm12,%xmm14 +vmovdqa %xmm5,544(%rsp) +vpmuludq v19_19(%rip),%xmm5,%xmm5 +vmovdqa %xmm5,560(%rsp) +vpaddq %xmm14,%xmm6,%xmm6 +vpmuludq %xmm7,%xmm10,%xmm5 +vmovdqa %xmm7,576(%rsp) +vpaddq %xmm7,%xmm7,%xmm7 +vpmuludq %xmm3,%xmm12,%xmm14 +vmovdqa %xmm3,592(%rsp) +vpaddq %xmm14,%xmm5,%xmm5 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vmovdqa %xmm3,608(%rsp) +vpaddq %xmm9,%xmm8,%xmm3 +vpaddq subc2(%rip),%xmm8,%xmm8 +vpsubq %xmm9,%xmm8,%xmm8 +vpunpckhqdq %xmm3,%xmm8,%xmm9 +vpunpcklqdq %xmm3,%xmm8,%xmm3 +vmovdqa %xmm3,624(%rsp) +vpmuludq %xmm7,%xmm12,%xmm8 +vmovdqa %xmm7,640(%rsp) +vpmuludq v19_19(%rip),%xmm7,%xmm7 +vmovdqa %xmm7,656(%rsp) +vpmuludq %xmm3,%xmm10,%xmm7 +vpaddq %xmm7,%xmm8,%xmm8 +vpmuludq %xmm9,%xmm10,%xmm7 +vmovdqa %xmm9,672(%rsp) +vpaddq %xmm9,%xmm9,%xmm9 +vpmuludq %xmm3,%xmm12,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vmovdqa %xmm3,688(%rsp) +vpmuludq v19_19(%rip),%xmm12,%xmm12 +vpmuludq %xmm9,%xmm12,%xmm3 +vmovdqa %xmm9,704(%rsp) +vpaddq %xmm3,%xmm11,%xmm11 +vmovdqa 0(%rsp),%xmm3 +vmovdqa 16(%rsp),%xmm9 +vpaddq subc2(%rip),%xmm3,%xmm10 +vpsubq %xmm9,%xmm10,%xmm10 +vpaddq %xmm9,%xmm3,%xmm3 +vpunpckhqdq %xmm10,%xmm3,%xmm9 +vpunpcklqdq %xmm10,%xmm3,%xmm3 +vpmuludq 144(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm0,%xmm0 +vpmuludq 128(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm2,%xmm2 +vpmuludq 480(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpmuludq 464(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm4,%xmm4 +vpmuludq 528(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm6,%xmm6 +vpmuludq 512(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm5,%xmm5 +vpmuludq 592(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpmuludq 576(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vpmuludq 624(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm11,%xmm11 +vpmuludq 672(%rsp),%xmm3,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 144(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpmuludq 448(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm1,%xmm1 +vpmuludq 480(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 496(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 528(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 544(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm8,%xmm8 +vpmuludq 592(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm9,%xmm9 +vpmuludq 640(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 624(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 704(%rsp),%xmm9,%xmm9 +vpaddq %xmm9,%xmm0,%xmm0 +vmovdqa 32(%rsp),%xmm3 +vmovdqa 80(%rsp),%xmm9 +vpaddq subc2(%rip),%xmm3,%xmm10 +vpsubq %xmm9,%xmm10,%xmm10 +vpaddq %xmm9,%xmm3,%xmm3 +vpunpckhqdq %xmm10,%xmm3,%xmm9 +vpunpcklqdq %xmm10,%xmm3,%xmm3 +vpmuludq 144(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpmuludq 128(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm4,%xmm4 +vpmuludq 480(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm6,%xmm6 +vpmuludq 464(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm5,%xmm5 +vpmuludq 528(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpmuludq 512(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vpmuludq 592(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm11,%xmm11 +vpmuludq 576(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm13,%xmm13 +vpmuludq 624(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm0,%xmm0 +vpmuludq 672(%rsp),%xmm3,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpmuludq 144(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 448(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 480(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 496(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm8,%xmm8 +vpmuludq 528(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm9,%xmm9 +vpmuludq 544(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 592(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 640(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm0,%xmm0 +vpmuludq 624(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpmuludq 704(%rsp),%xmm9,%xmm9 +vpaddq %xmm9,%xmm1,%xmm1 +vmovdqa 48(%rsp),%xmm3 +vmovdqa 96(%rsp),%xmm9 +vpaddq subc2(%rip),%xmm3,%xmm10 +vpsubq %xmm9,%xmm10,%xmm10 +vpaddq %xmm9,%xmm3,%xmm3 +vpunpckhqdq %xmm10,%xmm3,%xmm9 +vpunpcklqdq %xmm10,%xmm3,%xmm3 +vpmuludq 144(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm6,%xmm6 +vpmuludq 128(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm5,%xmm5 +vpmuludq 480(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpmuludq 464(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vpmuludq 528(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm11,%xmm11 +vpmuludq 512(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm13,%xmm13 +vpmuludq 592(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm0,%xmm0 +vpmuludq 576(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm2,%xmm2 +vpmuludq 624(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpmuludq 672(%rsp),%xmm3,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 144(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 448(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm8,%xmm8 +vpmuludq 480(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm9,%xmm9 +vpmuludq 496(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 528(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 544(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm0,%xmm0 +vpmuludq 592(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpmuludq 640(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm1,%xmm1 +vpmuludq 624(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 704(%rsp),%xmm9,%xmm9 +vpaddq %xmm9,%xmm6,%xmm6 +vmovdqa 64(%rsp),%xmm3 +vmovdqa 112(%rsp),%xmm9 +vpaddq subc2(%rip),%xmm3,%xmm10 +vpsubq %xmm9,%xmm10,%xmm10 +vpaddq %xmm9,%xmm3,%xmm3 +vpunpckhqdq %xmm10,%xmm3,%xmm9 +vpunpcklqdq %xmm10,%xmm3,%xmm3 +vpmuludq 144(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpmuludq 128(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vpmuludq 480(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm11,%xmm11 +vpmuludq 464(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm13,%xmm13 +vpmuludq 528(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm0,%xmm0 +vpmuludq 512(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm2,%xmm2 +vpmuludq 592(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpmuludq 576(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm4,%xmm4 +vpmuludq 624(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm6,%xmm6 +vpmuludq 672(%rsp),%xmm3,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 144(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm9,%xmm9 +vpmuludq 448(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 480(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 496(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm0,%xmm0 +vpmuludq 528(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpmuludq 544(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm1,%xmm1 +vpmuludq 592(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 640(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 624(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 704(%rsp),%xmm9,%xmm9 +vpaddq %xmm9,%xmm8,%xmm8 +vpsrlq $25,%xmm4,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpand m25(%rip),%xmm4,%xmm4 +vpsrlq $26,%xmm11,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpand m26(%rip),%xmm11,%xmm11 +vpsrlq $26,%xmm6,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpand m26(%rip),%xmm6,%xmm6 +vpsrlq $25,%xmm13,%xmm3 +vpaddq %xmm3,%xmm0,%xmm0 +vpand m25(%rip),%xmm13,%xmm13 +vpsrlq $25,%xmm5,%xmm3 +vpaddq %xmm3,%xmm8,%xmm8 +vpand m25(%rip),%xmm5,%xmm5 +vpsrlq $26,%xmm0,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpand m26(%rip),%xmm0,%xmm0 +vpsrlq $26,%xmm8,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpand m26(%rip),%xmm8,%xmm8 +vpsrlq $25,%xmm2,%xmm3 +vpaddq %xmm3,%xmm1,%xmm1 +vpand m25(%rip),%xmm2,%xmm2 +vpsrlq $25,%xmm7,%xmm3 +vpsllq $4,%xmm3,%xmm9 +vpaddq %xmm3,%xmm11,%xmm11 +vpsllq $1,%xmm3,%xmm3 +vpaddq %xmm3,%xmm9,%xmm9 +vpaddq %xmm9,%xmm11,%xmm11 +vpand m25(%rip),%xmm7,%xmm7 +vpsrlq $26,%xmm1,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpand m26(%rip),%xmm1,%xmm1 +vpsrlq $26,%xmm11,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpand m26(%rip),%xmm11,%xmm11 +vpsrlq $25,%xmm4,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpand m25(%rip),%xmm4,%xmm4 +vpunpcklqdq %xmm13,%xmm11,%xmm3 +vpunpckhqdq %xmm13,%xmm11,%xmm9 +vpaddq subc0(%rip),%xmm9,%xmm10 +vpsubq %xmm3,%xmm10,%xmm10 +vpaddq %xmm9,%xmm3,%xmm3 +vpunpckhqdq %xmm3,%xmm10,%xmm9 +vpunpcklqdq %xmm3,%xmm10,%xmm10 +vpmuludq %xmm10,%xmm10,%xmm3 +vpaddq %xmm10,%xmm10,%xmm10 +vpmuludq %xmm9,%xmm10,%xmm11 +vpunpcklqdq %xmm2,%xmm0,%xmm12 +vpunpckhqdq %xmm2,%xmm0,%xmm0 +vpaddq subc2(%rip),%xmm0,%xmm2 +vpsubq %xmm12,%xmm2,%xmm2 +vpaddq %xmm0,%xmm12,%xmm12 +vpunpckhqdq %xmm12,%xmm2,%xmm0 +vpunpcklqdq %xmm12,%xmm2,%xmm2 +vpmuludq %xmm2,%xmm10,%xmm12 +vpaddq %xmm9,%xmm9,%xmm13 +vpmuludq %xmm13,%xmm9,%xmm9 +vpaddq %xmm9,%xmm12,%xmm12 +vpmuludq %xmm0,%xmm10,%xmm9 +vpmuludq %xmm2,%xmm13,%xmm14 +vpaddq %xmm14,%xmm9,%xmm9 +vpunpcklqdq %xmm4,%xmm1,%xmm14 +vpunpckhqdq %xmm4,%xmm1,%xmm1 +vpaddq subc2(%rip),%xmm1,%xmm4 +vpsubq %xmm14,%xmm4,%xmm4 +vpaddq %xmm1,%xmm14,%xmm14 +vpunpckhqdq %xmm14,%xmm4,%xmm1 +vpunpcklqdq %xmm14,%xmm4,%xmm4 +vmovdqa %xmm1,0(%rsp) +vpaddq %xmm1,%xmm1,%xmm1 +vmovdqa %xmm1,16(%rsp) +vpmuludq v19_19(%rip),%xmm1,%xmm1 +vmovdqa %xmm1,32(%rsp) +vpmuludq %xmm4,%xmm10,%xmm1 +vpmuludq %xmm2,%xmm2,%xmm14 +vpaddq %xmm14,%xmm1,%xmm1 +vpmuludq 0(%rsp),%xmm10,%xmm14 +vpmuludq %xmm4,%xmm13,%xmm15 +vpaddq %xmm15,%xmm14,%xmm14 +vpunpcklqdq %xmm5,%xmm6,%xmm15 +vpunpckhqdq %xmm5,%xmm6,%xmm5 +vpaddq subc2(%rip),%xmm5,%xmm6 +vpsubq %xmm15,%xmm6,%xmm6 +vpaddq %xmm5,%xmm15,%xmm15 +vpunpckhqdq %xmm15,%xmm6,%xmm5 +vpunpcklqdq %xmm15,%xmm6,%xmm6 +vmovdqa %xmm6,48(%rsp) +vpmuludq v19_19(%rip),%xmm6,%xmm6 +vmovdqa %xmm6,64(%rsp) +vmovdqa %xmm5,80(%rsp) +vpmuludq v38_38(%rip),%xmm5,%xmm5 +vmovdqa %xmm5,96(%rsp) +vpmuludq 48(%rsp),%xmm10,%xmm5 +vpaddq %xmm0,%xmm0,%xmm6 +vpmuludq %xmm6,%xmm0,%xmm0 +vpaddq %xmm0,%xmm5,%xmm5 +vpmuludq 80(%rsp),%xmm10,%xmm0 +vpmuludq %xmm4,%xmm6,%xmm15 +vpaddq %xmm15,%xmm0,%xmm0 +vpmuludq %xmm6,%xmm13,%xmm15 +vpaddq %xmm15,%xmm1,%xmm1 +vpmuludq %xmm6,%xmm2,%xmm15 +vpaddq %xmm15,%xmm14,%xmm14 +vpunpcklqdq %xmm7,%xmm8,%xmm15 +vpunpckhqdq %xmm7,%xmm8,%xmm7 +vpaddq subc2(%rip),%xmm7,%xmm8 +vpsubq %xmm15,%xmm8,%xmm8 +vpaddq %xmm7,%xmm15,%xmm15 +vpunpckhqdq %xmm15,%xmm8,%xmm7 +vpunpcklqdq %xmm15,%xmm8,%xmm8 +vmovdqa %xmm8,112(%rsp) +vpmuludq v19_19(%rip),%xmm8,%xmm8 +vmovdqa %xmm8,448(%rsp) +vpmuludq 112(%rsp),%xmm10,%xmm8 +vpmuludq %xmm7,%xmm10,%xmm10 +vpmuludq v38_38(%rip),%xmm7,%xmm15 +vpmuludq %xmm15,%xmm7,%xmm7 +vpaddq %xmm7,%xmm8,%xmm8 +vpmuludq %xmm15,%xmm13,%xmm7 +vpaddq %xmm7,%xmm3,%xmm3 +vpmuludq %xmm15,%xmm2,%xmm7 +vpaddq %xmm7,%xmm11,%xmm11 +vpmuludq 80(%rsp),%xmm13,%xmm7 +vpaddq %xmm7,%xmm7,%xmm7 +vpaddq %xmm7,%xmm8,%xmm8 +vpmuludq 16(%rsp),%xmm13,%xmm7 +vpaddq %xmm7,%xmm5,%xmm5 +vpmuludq 48(%rsp),%xmm13,%xmm7 +vpaddq %xmm7,%xmm0,%xmm0 +vpmuludq 112(%rsp),%xmm13,%xmm7 +vpaddq %xmm7,%xmm10,%xmm10 +vpmuludq %xmm15,%xmm6,%xmm7 +vpaddq %xmm7,%xmm12,%xmm12 +vpmuludq %xmm15,%xmm4,%xmm7 +vpaddq %xmm7,%xmm9,%xmm9 +vpaddq %xmm2,%xmm2,%xmm2 +vpmuludq %xmm4,%xmm2,%xmm7 +vpaddq %xmm7,%xmm5,%xmm5 +vpmuludq 448(%rsp),%xmm2,%xmm7 +vpaddq %xmm7,%xmm3,%xmm3 +vpmuludq 448(%rsp),%xmm6,%xmm7 +vpaddq %xmm7,%xmm11,%xmm11 +vpmuludq 0(%rsp),%xmm2,%xmm7 +vpaddq %xmm7,%xmm0,%xmm0 +vpmuludq 48(%rsp),%xmm2,%xmm7 +vpaddq %xmm7,%xmm8,%xmm8 +vpmuludq 80(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 96(%rsp),%xmm4,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpmuludq %xmm4,%xmm4,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpaddq %xmm4,%xmm4,%xmm2 +vpmuludq 448(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm12,%xmm12 +vpmuludq 16(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vpmuludq 48(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm14,%xmm14 +vpmuludq 96(%rsp),%xmm6,%xmm4 +vpaddq %xmm4,%xmm3,%xmm3 +vmovdqa 16(%rsp),%xmm4 +vpmuludq 448(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm9,%xmm9 +vpmuludq 16(%rsp),%xmm6,%xmm4 +vpaddq %xmm4,%xmm8,%xmm8 +vpmuludq 48(%rsp),%xmm6,%xmm4 +vpaddq %xmm4,%xmm10,%xmm10 +vpmuludq 80(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm4,%xmm4 +vpaddq %xmm4,%xmm5,%xmm5 +vpmuludq 112(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm0,%xmm0 +vmovdqa 48(%rsp),%xmm4 +vpaddq %xmm4,%xmm4,%xmm4 +vpmuludq 448(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vmovdqa 80(%rsp),%xmm4 +vpaddq %xmm4,%xmm4,%xmm4 +vpmuludq 448(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm14,%xmm14 +vpmuludq 64(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm3,%xmm3 +vmovdqa 16(%rsp),%xmm4 +vpmuludq 64(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm11,%xmm11 +vmovdqa 16(%rsp),%xmm4 +vpmuludq 96(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm12,%xmm12 +vmovdqa 48(%rsp),%xmm4 +vpmuludq 96(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm9,%xmm9 +vpmuludq 0(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vmovdqa 32(%rsp),%xmm2 +vpmuludq 0(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm3,%xmm3 +vmovdqa 64(%rsp),%xmm2 +vpmuludq 48(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vmovdqa 96(%rsp),%xmm2 +vpmuludq 80(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm1,%xmm1 +vmovdqa 448(%rsp),%xmm2 +vpmuludq 112(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpsrlq $26,%xmm3,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpand m26(%rip),%xmm3,%xmm3 +vpsrlq $25,%xmm14,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpand m25(%rip),%xmm14,%xmm14 +vpsrlq $25,%xmm11,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpand m25(%rip),%xmm11,%xmm11 +vpsrlq $26,%xmm5,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vpand m26(%rip),%xmm5,%xmm5 +vpsrlq $26,%xmm12,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpand m26(%rip),%xmm12,%xmm12 +vpsrlq $25,%xmm0,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpand m25(%rip),%xmm0,%xmm0 +vpsrlq $25,%xmm9,%xmm2 +vpaddq %xmm2,%xmm1,%xmm1 +vpand m25(%rip),%xmm9,%xmm9 +vpsrlq $26,%xmm8,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpand m26(%rip),%xmm8,%xmm8 +vpsrlq $26,%xmm1,%xmm2 +vpaddq %xmm2,%xmm14,%xmm14 +vpand m26(%rip),%xmm1,%xmm1 +vpsrlq $25,%xmm10,%xmm2 +vpsllq $4,%xmm2,%xmm4 +vpaddq %xmm2,%xmm3,%xmm3 +vpsllq $1,%xmm2,%xmm2 +vpaddq %xmm2,%xmm4,%xmm4 +vpaddq %xmm4,%xmm3,%xmm3 +vpand m25(%rip),%xmm10,%xmm10 +vpsrlq $25,%xmm14,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpand m25(%rip),%xmm14,%xmm14 +vpsrlq $26,%xmm3,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpand m26(%rip),%xmm3,%xmm3 +vpunpckhqdq %xmm11,%xmm3,%xmm2 +vmovdqa %xmm2,0(%rsp) +vpshufd $0,%xmm3,%xmm2 +vpshufd $0,%xmm11,%xmm3 +vpmuludq 160(%rsp),%xmm2,%xmm4 +vpmuludq 432(%rsp),%xmm3,%xmm6 +vpaddq %xmm6,%xmm4,%xmm4 +vpmuludq 176(%rsp),%xmm2,%xmm6 +vpmuludq 304(%rsp),%xmm3,%xmm7 +vpaddq %xmm7,%xmm6,%xmm6 +vpmuludq 208(%rsp),%xmm2,%xmm7 +vpmuludq 336(%rsp),%xmm3,%xmm11 +vpaddq %xmm11,%xmm7,%xmm7 +vpmuludq 240(%rsp),%xmm2,%xmm11 +vpmuludq 368(%rsp),%xmm3,%xmm13 +vpaddq %xmm13,%xmm11,%xmm11 +vpmuludq 272(%rsp),%xmm2,%xmm2 +vpmuludq 400(%rsp),%xmm3,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpunpckhqdq %xmm9,%xmm12,%xmm3 +vmovdqa %xmm3,16(%rsp) +vpshufd $0,%xmm12,%xmm3 +vpshufd $0,%xmm9,%xmm9 +vpmuludq 288(%rsp),%xmm3,%xmm12 +vpaddq %xmm12,%xmm4,%xmm4 +vpmuludq 416(%rsp),%xmm9,%xmm12 +vpaddq %xmm12,%xmm4,%xmm4 +vpmuludq 160(%rsp),%xmm3,%xmm12 +vpaddq %xmm12,%xmm6,%xmm6 +vpmuludq 432(%rsp),%xmm9,%xmm12 +vpaddq %xmm12,%xmm6,%xmm6 +vpmuludq 176(%rsp),%xmm3,%xmm12 +vpaddq %xmm12,%xmm7,%xmm7 +vpmuludq 304(%rsp),%xmm9,%xmm12 +vpaddq %xmm12,%xmm7,%xmm7 +vpmuludq 208(%rsp),%xmm3,%xmm12 +vpaddq %xmm12,%xmm11,%xmm11 +vpmuludq 336(%rsp),%xmm9,%xmm12 +vpaddq %xmm12,%xmm11,%xmm11 +vpmuludq 240(%rsp),%xmm3,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpmuludq 368(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpunpckhqdq %xmm14,%xmm1,%xmm3 +vmovdqa %xmm3,32(%rsp) +vpshufd $0,%xmm1,%xmm1 +vpshufd $0,%xmm14,%xmm3 +vpmuludq 256(%rsp),%xmm1,%xmm9 +vpaddq %xmm9,%xmm4,%xmm4 +vpmuludq 384(%rsp),%xmm3,%xmm9 +vpaddq %xmm9,%xmm4,%xmm4 +vpmuludq 288(%rsp),%xmm1,%xmm9 +vpaddq %xmm9,%xmm6,%xmm6 +vpmuludq 416(%rsp),%xmm3,%xmm9 +vpaddq %xmm9,%xmm6,%xmm6 +vpmuludq 160(%rsp),%xmm1,%xmm9 +vpaddq %xmm9,%xmm7,%xmm7 +vpmuludq 432(%rsp),%xmm3,%xmm9 +vpaddq %xmm9,%xmm7,%xmm7 +vpmuludq 176(%rsp),%xmm1,%xmm9 +vpaddq %xmm9,%xmm11,%xmm11 +vpmuludq 304(%rsp),%xmm3,%xmm9 +vpaddq %xmm9,%xmm11,%xmm11 +vpmuludq 208(%rsp),%xmm1,%xmm1 +vpaddq %xmm1,%xmm2,%xmm2 +vpmuludq 336(%rsp),%xmm3,%xmm1 +vpaddq %xmm1,%xmm2,%xmm2 +vpunpckhqdq %xmm0,%xmm5,%xmm1 +vmovdqa %xmm1,48(%rsp) +vpshufd $0,%xmm5,%xmm1 +vpshufd $0,%xmm0,%xmm0 +vpmuludq 224(%rsp),%xmm1,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 352(%rsp),%xmm0,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 256(%rsp),%xmm1,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 384(%rsp),%xmm0,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 288(%rsp),%xmm1,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq 416(%rsp),%xmm0,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq 160(%rsp),%xmm1,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 432(%rsp),%xmm0,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 176(%rsp),%xmm1,%xmm1 +vpaddq %xmm1,%xmm2,%xmm2 +vpmuludq 304(%rsp),%xmm0,%xmm0 +vpaddq %xmm0,%xmm2,%xmm2 +vpunpckhqdq %xmm10,%xmm8,%xmm0 +vmovdqa %xmm0,64(%rsp) +vpshufd $0,%xmm8,%xmm0 +vpshufd $0,%xmm10,%xmm1 +vpmuludq 192(%rsp),%xmm0,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 320(%rsp),%xmm1,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 224(%rsp),%xmm0,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 352(%rsp),%xmm1,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 256(%rsp),%xmm0,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq 384(%rsp),%xmm1,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq 288(%rsp),%xmm0,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 416(%rsp),%xmm1,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 160(%rsp),%xmm0,%xmm0 +vpaddq %xmm0,%xmm2,%xmm2 +vpmuludq 432(%rsp),%xmm1,%xmm0 +vpaddq %xmm0,%xmm2,%xmm2 +vmovdqa %xmm4,80(%rsp) +vmovdqa %xmm6,96(%rsp) +vmovdqa %xmm7,112(%rsp) +vmovdqa %xmm11,448(%rsp) +vmovdqa %xmm2,496(%rsp) +vmovdqa 144(%rsp),%xmm0 +vpmuludq %xmm0,%xmm0,%xmm1 +vpaddq %xmm0,%xmm0,%xmm0 +vmovdqa 128(%rsp),%xmm2 +vpmuludq %xmm2,%xmm0,%xmm3 +vmovdqa 480(%rsp),%xmm4 +vpmuludq %xmm4,%xmm0,%xmm5 +vmovdqa 464(%rsp),%xmm6 +vpmuludq %xmm6,%xmm0,%xmm7 +vmovdqa 528(%rsp),%xmm8 +vpmuludq %xmm8,%xmm0,%xmm9 +vpmuludq 512(%rsp),%xmm0,%xmm10 +vpmuludq 592(%rsp),%xmm0,%xmm11 +vpmuludq 576(%rsp),%xmm0,%xmm12 +vpmuludq 624(%rsp),%xmm0,%xmm13 +vmovdqa 672(%rsp),%xmm14 +vpmuludq %xmm14,%xmm0,%xmm0 +vpmuludq v38_38(%rip),%xmm14,%xmm15 +vpmuludq %xmm15,%xmm14,%xmm14 +vpaddq %xmm14,%xmm13,%xmm13 +vpaddq %xmm6,%xmm6,%xmm14 +vpmuludq %xmm14,%xmm6,%xmm6 +vpaddq %xmm6,%xmm11,%xmm11 +vpaddq %xmm2,%xmm2,%xmm6 +vpmuludq %xmm6,%xmm2,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpmuludq %xmm15,%xmm6,%xmm2 +vpaddq %xmm2,%xmm1,%xmm1 +vpmuludq %xmm15,%xmm4,%xmm2 +vpaddq %xmm2,%xmm3,%xmm3 +vpmuludq 544(%rsp),%xmm6,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpmuludq 592(%rsp),%xmm6,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 640(%rsp),%xmm6,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpmuludq 624(%rsp),%xmm6,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vpmuludq %xmm4,%xmm6,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpmuludq %xmm14,%xmm6,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpmuludq %xmm8,%xmm6,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq %xmm15,%xmm14,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpmuludq %xmm15,%xmm8,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpmuludq %xmm4,%xmm4,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpmuludq %xmm14,%xmm4,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpaddq %xmm4,%xmm4,%xmm2 +vpmuludq %xmm8,%xmm2,%xmm4 +vpaddq %xmm4,%xmm11,%xmm11 +vpmuludq 688(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vpmuludq 688(%rsp),%xmm14,%xmm4 +vpaddq %xmm4,%xmm3,%xmm3 +vpmuludq 512(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm12,%xmm12 +vpmuludq 592(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm13,%xmm13 +vpmuludq 576(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vpmuludq 656(%rsp),%xmm8,%xmm2 +vpaddq %xmm2,%xmm3,%xmm3 +vpmuludq %xmm8,%xmm14,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq %xmm8,%xmm8,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpaddq %xmm8,%xmm8,%xmm2 +vpmuludq 688(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm5,%xmm5 +vpmuludq 544(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm9,%xmm9 +vpmuludq 592(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm10,%xmm10 +vpmuludq 656(%rsp),%xmm14,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vmovdqa 544(%rsp),%xmm4 +vpmuludq 688(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm7,%xmm7 +vpmuludq 544(%rsp),%xmm14,%xmm4 +vpaddq %xmm4,%xmm13,%xmm13 +vpmuludq 592(%rsp),%xmm14,%xmm4 +vpaddq %xmm4,%xmm0,%xmm0 +vpmuludq 640(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm11,%xmm11 +vpmuludq 624(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm12,%xmm12 +vmovdqa 592(%rsp),%xmm4 +vpaddq %xmm4,%xmm4,%xmm4 +vpmuludq 688(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm9,%xmm9 +vpmuludq 608(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vmovdqa 544(%rsp),%xmm4 +vpmuludq 608(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm3,%xmm3 +vmovdqa 544(%rsp),%xmm4 +vpmuludq 656(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm5,%xmm5 +vmovdqa 592(%rsp),%xmm4 +vpmuludq 656(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm7,%xmm7 +vmovdqa 640(%rsp),%xmm4 +vpmuludq 688(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm10,%xmm10 +vpmuludq 512(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vmovdqa 560(%rsp),%xmm2 +vpmuludq 512(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm1,%xmm1 +vmovdqa 608(%rsp),%xmm2 +vpmuludq 592(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vmovdqa 656(%rsp),%xmm2 +vpmuludq 576(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vmovdqa 688(%rsp),%xmm2 +vpmuludq 624(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpsrlq $26,%xmm1,%xmm2 +vpaddq %xmm2,%xmm3,%xmm3 +vpand m26(%rip),%xmm1,%xmm1 +vpsrlq $25,%xmm10,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpand m25(%rip),%xmm10,%xmm10 +vpsrlq $25,%xmm3,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpand m25(%rip),%xmm3,%xmm3 +vpsrlq $26,%xmm11,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpand m26(%rip),%xmm11,%xmm11 +vpsrlq $26,%xmm5,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpand m26(%rip),%xmm5,%xmm5 +vpsrlq $25,%xmm12,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpand m25(%rip),%xmm12,%xmm12 +vpsrlq $25,%xmm7,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpand m25(%rip),%xmm7,%xmm7 +vpsrlq $26,%xmm13,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vpand m26(%rip),%xmm13,%xmm13 +vpsrlq $26,%xmm9,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpand m26(%rip),%xmm9,%xmm9 +vpsrlq $25,%xmm0,%xmm2 +vpsllq $4,%xmm2,%xmm4 +vpaddq %xmm2,%xmm1,%xmm1 +vpsllq $1,%xmm2,%xmm2 +vpaddq %xmm2,%xmm4,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vpand m25(%rip),%xmm0,%xmm0 +vpsrlq $25,%xmm10,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpand m25(%rip),%xmm10,%xmm10 +vpsrlq $26,%xmm1,%xmm2 +vpaddq %xmm2,%xmm3,%xmm3 +vpand m26(%rip),%xmm1,%xmm1 +vpunpckhqdq %xmm3,%xmm1,%xmm2 +vpunpcklqdq %xmm3,%xmm1,%xmm1 +vmovdqa %xmm1,464(%rsp) +vpaddq subc0(%rip),%xmm2,%xmm3 +vpsubq %xmm1,%xmm3,%xmm3 +vpunpckhqdq %xmm3,%xmm2,%xmm1 +vpunpcklqdq %xmm3,%xmm2,%xmm2 +vmovdqa %xmm2,480(%rsp) +vmovdqa %xmm1,512(%rsp) +vpsllq $1,%xmm1,%xmm1 +vmovdqa %xmm1,528(%rsp) +vpmuludq v121666_121666(%rip),%xmm3,%xmm3 +vmovdqa 80(%rsp),%xmm1 +vpunpcklqdq %xmm1,%xmm3,%xmm2 +vpunpckhqdq %xmm1,%xmm3,%xmm1 +vpunpckhqdq %xmm7,%xmm5,%xmm3 +vpunpcklqdq %xmm7,%xmm5,%xmm4 +vmovdqa %xmm4,544(%rsp) +vpaddq subc2(%rip),%xmm3,%xmm5 +vpsubq %xmm4,%xmm5,%xmm5 +vpunpckhqdq %xmm5,%xmm3,%xmm4 +vpunpcklqdq %xmm5,%xmm3,%xmm3 +vmovdqa %xmm3,560(%rsp) +vmovdqa %xmm4,576(%rsp) +vpsllq $1,%xmm4,%xmm4 +vmovdqa %xmm4,592(%rsp) +vpmuludq v121666_121666(%rip),%xmm5,%xmm5 +vmovdqa 96(%rsp),%xmm3 +vpunpcklqdq %xmm3,%xmm5,%xmm4 +vpunpckhqdq %xmm3,%xmm5,%xmm3 +vpunpckhqdq %xmm10,%xmm9,%xmm5 +vpunpcklqdq %xmm10,%xmm9,%xmm6 +vmovdqa %xmm6,608(%rsp) +vpaddq subc2(%rip),%xmm5,%xmm7 +vpsubq %xmm6,%xmm7,%xmm7 +vpunpckhqdq %xmm7,%xmm5,%xmm6 +vpunpcklqdq %xmm7,%xmm5,%xmm5 +vmovdqa %xmm5,624(%rsp) +vmovdqa %xmm6,640(%rsp) +vpsllq $1,%xmm6,%xmm6 +vmovdqa %xmm6,656(%rsp) +vpmuludq v121666_121666(%rip),%xmm7,%xmm7 +vmovdqa 112(%rsp),%xmm5 +vpunpcklqdq %xmm5,%xmm7,%xmm6 +vpunpckhqdq %xmm5,%xmm7,%xmm5 +vpunpckhqdq %xmm12,%xmm11,%xmm7 +vpunpcklqdq %xmm12,%xmm11,%xmm8 +vmovdqa %xmm8,672(%rsp) +vpaddq subc2(%rip),%xmm7,%xmm9 +vpsubq %xmm8,%xmm9,%xmm9 +vpunpckhqdq %xmm9,%xmm7,%xmm8 +vpunpcklqdq %xmm9,%xmm7,%xmm7 +vmovdqa %xmm7,688(%rsp) +vmovdqa %xmm8,704(%rsp) +vpsllq $1,%xmm8,%xmm8 +vmovdqa %xmm8,720(%rsp) +vpmuludq v121666_121666(%rip),%xmm9,%xmm9 +vmovdqa 448(%rsp),%xmm7 +vpunpcklqdq %xmm7,%xmm9,%xmm8 +vpunpckhqdq %xmm7,%xmm9,%xmm7 +vpunpckhqdq %xmm0,%xmm13,%xmm9 +vpunpcklqdq %xmm0,%xmm13,%xmm0 +vmovdqa %xmm0,448(%rsp) +vpaddq subc2(%rip),%xmm9,%xmm10 +vpsubq %xmm0,%xmm10,%xmm10 +vpunpckhqdq %xmm10,%xmm9,%xmm0 +vpunpcklqdq %xmm10,%xmm9,%xmm9 +vmovdqa %xmm9,736(%rsp) +vmovdqa %xmm0,752(%rsp) +vpsllq $1,%xmm0,%xmm0 +vmovdqa %xmm0,768(%rsp) +vpmuludq v121666_121666(%rip),%xmm10,%xmm10 +vmovdqa 496(%rsp),%xmm0 +vpunpcklqdq %xmm0,%xmm10,%xmm9 +vpunpckhqdq %xmm0,%xmm10,%xmm0 +vpsrlq $26,%xmm2,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpand m26(%rip),%xmm2,%xmm2 +vpsrlq $25,%xmm5,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpand m25(%rip),%xmm5,%xmm5 +vpsrlq $25,%xmm1,%xmm10 +vpaddq %xmm10,%xmm4,%xmm4 +vpand m25(%rip),%xmm1,%xmm1 +vpsrlq $26,%xmm8,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpand m26(%rip),%xmm8,%xmm8 +vpsrlq $26,%xmm4,%xmm10 +vpaddq %xmm10,%xmm3,%xmm3 +vpand m26(%rip),%xmm4,%xmm4 +vpsrlq $25,%xmm7,%xmm10 +vpaddq %xmm10,%xmm9,%xmm9 +vpand m25(%rip),%xmm7,%xmm7 +vpsrlq $25,%xmm3,%xmm10 +vpaddq %xmm10,%xmm6,%xmm6 +vpand m25(%rip),%xmm3,%xmm3 +vpsrlq $26,%xmm9,%xmm10 +vpaddq %xmm10,%xmm0,%xmm0 +vpand m26(%rip),%xmm9,%xmm9 +vpsrlq $26,%xmm6,%xmm10 +vpaddq %xmm10,%xmm5,%xmm5 +vpand m26(%rip),%xmm6,%xmm6 +vpsrlq $25,%xmm0,%xmm10 +vpsllq $4,%xmm10,%xmm11 +vpaddq %xmm10,%xmm2,%xmm2 +vpsllq $1,%xmm10,%xmm10 +vpaddq %xmm10,%xmm11,%xmm11 +vpaddq %xmm11,%xmm2,%xmm2 +vpand m25(%rip),%xmm0,%xmm0 +vpsrlq $25,%xmm5,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpand m25(%rip),%xmm5,%xmm5 +vpsrlq $26,%xmm2,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpand m26(%rip),%xmm2,%xmm2 +vpunpckhqdq %xmm1,%xmm2,%xmm10 +vmovdqa %xmm10,80(%rsp) +vpunpcklqdq %xmm1,%xmm2,%xmm1 +vpunpckhqdq %xmm3,%xmm4,%xmm2 +vmovdqa %xmm2,96(%rsp) +vpunpcklqdq %xmm3,%xmm4,%xmm2 +vpunpckhqdq %xmm5,%xmm6,%xmm3 +vmovdqa %xmm3,112(%rsp) +vpunpcklqdq %xmm5,%xmm6,%xmm3 +vpunpckhqdq %xmm7,%xmm8,%xmm4 +vmovdqa %xmm4,128(%rsp) +vpunpcklqdq %xmm7,%xmm8,%xmm4 +vpunpckhqdq %xmm0,%xmm9,%xmm5 +vmovdqa %xmm5,144(%rsp) +vpunpcklqdq %xmm0,%xmm9,%xmm0 +vmovdqa 464(%rsp),%xmm5 +vpaddq %xmm5,%xmm1,%xmm1 +vpunpcklqdq %xmm1,%xmm5,%xmm6 +vpunpckhqdq %xmm1,%xmm5,%xmm1 +vpmuludq 512(%rsp),%xmm6,%xmm5 +vpmuludq 480(%rsp),%xmm1,%xmm7 +vpaddq %xmm7,%xmm5,%xmm5 +vpmuludq 560(%rsp),%xmm6,%xmm7 +vpmuludq 528(%rsp),%xmm1,%xmm8 +vpaddq %xmm8,%xmm7,%xmm7 +vpmuludq 576(%rsp),%xmm6,%xmm8 +vpmuludq 560(%rsp),%xmm1,%xmm9 +vpaddq %xmm9,%xmm8,%xmm8 +vpmuludq 624(%rsp),%xmm6,%xmm9 +vpmuludq 592(%rsp),%xmm1,%xmm10 +vpaddq %xmm10,%xmm9,%xmm9 +vpmuludq 640(%rsp),%xmm6,%xmm10 +vpmuludq 624(%rsp),%xmm1,%xmm11 +vpaddq %xmm11,%xmm10,%xmm10 +vpmuludq 688(%rsp),%xmm6,%xmm11 +vpmuludq 656(%rsp),%xmm1,%xmm12 +vpaddq %xmm12,%xmm11,%xmm11 +vpmuludq 704(%rsp),%xmm6,%xmm12 +vpmuludq 688(%rsp),%xmm1,%xmm13 +vpaddq %xmm13,%xmm12,%xmm12 +vpmuludq 736(%rsp),%xmm6,%xmm13 +vpmuludq 720(%rsp),%xmm1,%xmm14 +vpaddq %xmm14,%xmm13,%xmm13 +vpmuludq 752(%rsp),%xmm6,%xmm14 +vpmuludq 736(%rsp),%xmm1,%xmm15 +vpaddq %xmm15,%xmm14,%xmm14 +vpmuludq 480(%rsp),%xmm6,%xmm6 +vpmuludq v19_19(%rip),%xmm1,%xmm1 +vpmuludq 768(%rsp),%xmm1,%xmm1 +vpaddq %xmm1,%xmm6,%xmm6 +vmovdqa 544(%rsp),%xmm1 +vpaddq %xmm1,%xmm2,%xmm2 +vpunpcklqdq %xmm2,%xmm1,%xmm15 +vpunpckhqdq %xmm2,%xmm1,%xmm1 +vpmuludq 480(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpmuludq 512(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpmuludq 560(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpmuludq 576(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 624(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpmuludq 640(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 688(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpmuludq 704(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm15,%xmm15 +vpmuludq 736(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm6,%xmm6 +vpmuludq 752(%rsp),%xmm15,%xmm15 +vpaddq %xmm15,%xmm5,%xmm5 +vpmuludq 480(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpmuludq 528(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpmuludq 560(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 592(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpmuludq 624(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 656(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpmuludq 688(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm1,%xmm1 +vpmuludq 720(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm6,%xmm6 +vpmuludq 736(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpmuludq 768(%rsp),%xmm1,%xmm1 +vpaddq %xmm1,%xmm7,%xmm7 +vmovdqa 608(%rsp),%xmm1 +vpaddq %xmm1,%xmm3,%xmm3 +vpunpcklqdq %xmm3,%xmm1,%xmm2 +vpunpckhqdq %xmm3,%xmm1,%xmm1 +vpmuludq 480(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm9,%xmm9 +vpmuludq 512(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm10,%xmm10 +vpmuludq 560(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 576(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm12,%xmm12 +vpmuludq 624(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 640(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm2,%xmm2 +vpmuludq 688(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 704(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 736(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq 752(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpmuludq 480(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 528(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpmuludq 560(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 592(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpmuludq 624(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm1,%xmm1 +vpmuludq 656(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm6,%xmm6 +vpmuludq 688(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpmuludq 720(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpmuludq 736(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpmuludq 768(%rsp),%xmm1,%xmm1 +vpaddq %xmm1,%xmm9,%xmm9 +vmovdqa 672(%rsp),%xmm1 +vpaddq %xmm1,%xmm4,%xmm4 +vpunpcklqdq %xmm4,%xmm1,%xmm2 +vpunpckhqdq %xmm4,%xmm1,%xmm1 +vpmuludq 480(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 512(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm12,%xmm12 +vpmuludq 560(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 576(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm2,%xmm2 +vpmuludq 624(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 640(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 688(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq 704(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm8,%xmm8 +vpmuludq 736(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm9,%xmm9 +vpmuludq 752(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 480(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 528(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpmuludq 560(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm1,%xmm1 +vpmuludq 592(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm6,%xmm6 +vpmuludq 624(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpmuludq 656(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpmuludq 688(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpmuludq 720(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpmuludq 736(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 768(%rsp),%xmm1,%xmm1 +vpaddq %xmm1,%xmm11,%xmm11 +vmovdqa 448(%rsp),%xmm1 +vpaddq %xmm1,%xmm0,%xmm0 +vpunpcklqdq %xmm0,%xmm1,%xmm2 +vpunpckhqdq %xmm0,%xmm1,%xmm0 +vpmuludq 480(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm13,%xmm13 +vpmuludq 512(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm2,%xmm2 +vpmuludq 560(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm6,%xmm6 +vpmuludq 576(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm5,%xmm5 +vpmuludq 624(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm7,%xmm7 +vpmuludq 640(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm8,%xmm8 +vpmuludq 688(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm9,%xmm9 +vpmuludq 704(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm10,%xmm10 +vpmuludq 736(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm11,%xmm11 +vpmuludq 752(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 480(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm0,%xmm0 +vpmuludq 528(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm6,%xmm6 +vpmuludq 560(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm5,%xmm5 +vpmuludq 592(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm7,%xmm7 +vpmuludq 624(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm8,%xmm8 +vpmuludq 656(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm9,%xmm9 +vpmuludq 688(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm10,%xmm10 +vpmuludq 720(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm11,%xmm11 +vpmuludq 736(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm12,%xmm12 +vpmuludq 768(%rsp),%xmm0,%xmm0 +vpaddq %xmm0,%xmm13,%xmm13 +vpsrlq $26,%xmm6,%xmm0 +vpaddq %xmm0,%xmm5,%xmm5 +vpand m26(%rip),%xmm6,%xmm6 +vpsrlq $25,%xmm10,%xmm0 +vpaddq %xmm0,%xmm11,%xmm11 +vpand m25(%rip),%xmm10,%xmm10 +vpsrlq $25,%xmm5,%xmm0 +vpaddq %xmm0,%xmm7,%xmm7 +vpand m25(%rip),%xmm5,%xmm5 +vpsrlq $26,%xmm11,%xmm0 +vpaddq %xmm0,%xmm12,%xmm12 +vpand m26(%rip),%xmm11,%xmm11 +vpsrlq $26,%xmm7,%xmm0 +vpaddq %xmm0,%xmm8,%xmm8 +vpand m26(%rip),%xmm7,%xmm7 +vpsrlq $25,%xmm12,%xmm0 +vpaddq %xmm0,%xmm13,%xmm13 +vpand m25(%rip),%xmm12,%xmm12 +vpsrlq $25,%xmm8,%xmm0 +vpaddq %xmm0,%xmm9,%xmm9 +vpand m25(%rip),%xmm8,%xmm8 +vpsrlq $26,%xmm13,%xmm0 +vpaddq %xmm0,%xmm14,%xmm14 +vpand m26(%rip),%xmm13,%xmm13 +vpsrlq $26,%xmm9,%xmm0 +vpaddq %xmm0,%xmm10,%xmm10 +vpand m26(%rip),%xmm9,%xmm9 +vpsrlq $25,%xmm14,%xmm0 +vpsllq $4,%xmm0,%xmm1 +vpaddq %xmm0,%xmm6,%xmm6 +vpsllq $1,%xmm0,%xmm0 +vpaddq %xmm0,%xmm1,%xmm1 +vpaddq %xmm1,%xmm6,%xmm6 +vpand m25(%rip),%xmm14,%xmm14 +vpsrlq $25,%xmm10,%xmm0 +vpaddq %xmm0,%xmm11,%xmm11 +vpand m25(%rip),%xmm10,%xmm10 +vpsrlq $26,%xmm6,%xmm0 +vpaddq %xmm0,%xmm5,%xmm5 +vpand m26(%rip),%xmm6,%xmm6 +vpunpckhqdq %xmm5,%xmm6,%xmm1 +vpunpcklqdq %xmm5,%xmm6,%xmm0 +vpunpckhqdq %xmm8,%xmm7,%xmm3 +vpunpcklqdq %xmm8,%xmm7,%xmm2 +vpunpckhqdq %xmm10,%xmm9,%xmm5 +vpunpcklqdq %xmm10,%xmm9,%xmm4 +vpunpckhqdq %xmm12,%xmm11,%xmm7 +vpunpcklqdq %xmm12,%xmm11,%xmm6 +vpunpckhqdq %xmm14,%xmm13,%xmm9 +vpunpcklqdq %xmm14,%xmm13,%xmm8 +cmp $0,%rdx +jne ._ladder_loop +vmovdqu %xmm1,160(%rdi) +vmovdqu %xmm0,80(%rdi) +vmovdqu %xmm3,176(%rdi) +vmovdqu %xmm2,96(%rdi) +vmovdqu %xmm5,192(%rdi) +vmovdqu %xmm4,112(%rdi) +vmovdqu %xmm7,208(%rdi) +vmovdqu %xmm6,128(%rdi) +vmovdqu %xmm9,224(%rdi) +vmovdqu %xmm8,144(%rdi) +movq 1824(%rsp),%r11 +movq 1832(%rsp),%r12 +movq 1840(%rsp),%r13 +movq 1848(%rsp),%r14 +add %r11,%rsp +ret + +#endif + +.globl ladder_S_dummy diff --git a/sodium/sodium_sandy2x_ladder.h b/sodium/sodium_sandy2x_ladder.h new file mode 100644 index 00000000..bda7e7e5 --- /dev/null +++ b/sodium/sodium_sandy2x_ladder.h @@ -0,0 +1,18 @@ +#ifndef ladder_H +#define ladder_H + +#ifdef __cplusplus +extern "C" { +#endif + +#include "sodium_sandy2x_fe.h" +#include "sodium_sandy2x_ladder_namespace.h" + +extern void ladder(fe *, const unsigned char *); + +#ifdef __cplusplus +} +#endif + +#endif /* ifndef ladder_H */ + diff --git a/sodium/sodium_sandy2x_ladder_base.S b/sodium/sodium_sandy2x_ladder_base.S new file mode 100644 index 00000000..46c5ed38 --- /dev/null +++ b/sodium/sodium_sandy2x_ladder_base.S @@ -0,0 +1,1297 @@ +#ifdef IN_SANDY2X + +#include "sodium_sandy2x_ladder_base_namespace.h" +#include "sodium_sandy2x_consts_namespace.h" +.p2align 5 + +#ifdef ASM_HIDE_SYMBOL +ASM_HIDE_SYMBOL ladder_base +ASM_HIDE_SYMBOL _ladder_base +#endif +.globl ladder_base +.globl _ladder_base +#ifdef __ELF__ +.type ladder_base, @function +.type _ladder_base, @function +#endif +ladder_base: +_ladder_base: + +mov %rsp,%r11 +and $31,%r11 +add $1568,%r11 +sub %r11,%rsp +movq %r11,1536(%rsp) +movq %r12,1544(%rsp) +movq %r13,1552(%rsp) +vmovdqa v0_0(%rip),%xmm0 +vmovdqa v1_0(%rip),%xmm1 +vmovdqa v9_0(%rip),%xmm2 +vmovdqa %xmm2,0(%rsp) +vmovdqa %xmm0,16(%rsp) +vmovdqa %xmm0,32(%rsp) +vmovdqa %xmm0,48(%rsp) +vmovdqa %xmm0,64(%rsp) +vmovdqa %xmm1,80(%rsp) +vmovdqa %xmm0,96(%rsp) +vmovdqa %xmm0,112(%rsp) +vmovdqa %xmm0,128(%rsp) +vmovdqa %xmm0,144(%rsp) +vmovdqa %xmm1,%xmm0 +vpxor %xmm1,%xmm1,%xmm1 +vpxor %xmm2,%xmm2,%xmm2 +vpxor %xmm3,%xmm3,%xmm3 +vpxor %xmm4,%xmm4,%xmm4 +vpxor %xmm5,%xmm5,%xmm5 +vpxor %xmm6,%xmm6,%xmm6 +vpxor %xmm7,%xmm7,%xmm7 +vpxor %xmm8,%xmm8,%xmm8 +vpxor %xmm9,%xmm9,%xmm9 +movq 0(%rsi),%rdx +movq 8(%rsi),%rcx +movq 16(%rsi),%r8 +movq 24(%rsi),%r9 +shrd $1,%rcx,%rdx +shrd $1,%r8,%rcx +shrd $1,%r9,%r8 +shr $1,%r9 +xorq 0(%rsi),%rdx +xorq 8(%rsi),%rcx +xorq 16(%rsi),%r8 +xorq 24(%rsi),%r9 +leaq 512(%rsp),%rsi +mov $64,%rax + +.p2align 4 +._ladder_base_small_loop: +mov %rdx,%r10 +mov %rcx,%r11 +mov %r8,%r12 +mov %r9,%r13 +shr $1,%rdx +shr $1,%rcx +shr $1,%r8 +shr $1,%r9 +and $1,%r10d +and $1,%r11d +and $1,%r12d +and $1,%r13d +neg %r10 +neg %r11 +neg %r12 +neg %r13 +movl %r10d,0(%rsi) +movl %r11d,256(%rsi) +movl %r12d,512(%rsi) +movl %r13d,768(%rsi) +add $4,%rsi +sub $1,%rax +jne ._ladder_base_small_loop +mov $255,%rdx +add $760,%rsi + +.p2align 4 +._ladder_base_loop: +sub $1,%rdx +vbroadcastss 0(%rsi),%xmm10 +sub $4,%rsi +vmovdqa 0(%rsp),%xmm11 +vmovdqa 80(%rsp),%xmm12 +vpxor %xmm11,%xmm0,%xmm13 +vpand %xmm10,%xmm13,%xmm13 +vpxor %xmm13,%xmm0,%xmm0 +vpxor %xmm13,%xmm11,%xmm11 +vpxor %xmm12,%xmm1,%xmm13 +vpand %xmm10,%xmm13,%xmm13 +vpxor %xmm13,%xmm1,%xmm1 +vpxor %xmm13,%xmm12,%xmm12 +vmovdqa 16(%rsp),%xmm13 +vmovdqa 96(%rsp),%xmm14 +vpxor %xmm13,%xmm2,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm2,%xmm2 +vpxor %xmm15,%xmm13,%xmm13 +vpxor %xmm14,%xmm3,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm3,%xmm3 +vpxor %xmm15,%xmm14,%xmm14 +vmovdqa %xmm13,0(%rsp) +vmovdqa %xmm14,16(%rsp) +vmovdqa 32(%rsp),%xmm13 +vmovdqa 112(%rsp),%xmm14 +vpxor %xmm13,%xmm4,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm4,%xmm4 +vpxor %xmm15,%xmm13,%xmm13 +vpxor %xmm14,%xmm5,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm5,%xmm5 +vpxor %xmm15,%xmm14,%xmm14 +vmovdqa %xmm13,32(%rsp) +vmovdqa %xmm14,80(%rsp) +vmovdqa 48(%rsp),%xmm13 +vmovdqa 128(%rsp),%xmm14 +vpxor %xmm13,%xmm6,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm6,%xmm6 +vpxor %xmm15,%xmm13,%xmm13 +vpxor %xmm14,%xmm7,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm7,%xmm7 +vpxor %xmm15,%xmm14,%xmm14 +vmovdqa %xmm13,48(%rsp) +vmovdqa %xmm14,96(%rsp) +vmovdqa 64(%rsp),%xmm13 +vmovdqa 144(%rsp),%xmm14 +vpxor %xmm13,%xmm8,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm8,%xmm8 +vpxor %xmm15,%xmm13,%xmm13 +vpxor %xmm14,%xmm9,%xmm15 +vpand %xmm10,%xmm15,%xmm15 +vpxor %xmm15,%xmm9,%xmm9 +vpxor %xmm15,%xmm14,%xmm14 +vmovdqa %xmm13,64(%rsp) +vmovdqa %xmm14,112(%rsp) +vpaddq subc0(%rip),%xmm11,%xmm10 +vpsubq %xmm12,%xmm10,%xmm10 +vpaddq %xmm12,%xmm11,%xmm11 +vpunpckhqdq %xmm10,%xmm11,%xmm12 +vpunpcklqdq %xmm10,%xmm11,%xmm10 +vpaddq %xmm1,%xmm0,%xmm11 +vpaddq subc0(%rip),%xmm0,%xmm0 +vpsubq %xmm1,%xmm0,%xmm0 +vpunpckhqdq %xmm11,%xmm0,%xmm1 +vpunpcklqdq %xmm11,%xmm0,%xmm0 +vpmuludq %xmm0,%xmm10,%xmm11 +vpmuludq %xmm1,%xmm10,%xmm13 +vmovdqa %xmm1,128(%rsp) +vpaddq %xmm1,%xmm1,%xmm1 +vpmuludq %xmm0,%xmm12,%xmm14 +vmovdqa %xmm0,144(%rsp) +vpaddq %xmm14,%xmm13,%xmm13 +vpmuludq %xmm1,%xmm12,%xmm0 +vmovdqa %xmm1,160(%rsp) +vpaddq %xmm3,%xmm2,%xmm1 +vpaddq subc2(%rip),%xmm2,%xmm2 +vpsubq %xmm3,%xmm2,%xmm2 +vpunpckhqdq %xmm1,%xmm2,%xmm3 +vpunpcklqdq %xmm1,%xmm2,%xmm1 +vpmuludq %xmm1,%xmm10,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vpmuludq %xmm3,%xmm10,%xmm2 +vmovdqa %xmm3,176(%rsp) +vpaddq %xmm3,%xmm3,%xmm3 +vpmuludq %xmm1,%xmm12,%xmm14 +vmovdqa %xmm1,192(%rsp) +vpaddq %xmm14,%xmm2,%xmm2 +vpmuludq %xmm3,%xmm12,%xmm1 +vmovdqa %xmm3,208(%rsp) +vpaddq %xmm5,%xmm4,%xmm3 +vpaddq subc2(%rip),%xmm4,%xmm4 +vpsubq %xmm5,%xmm4,%xmm4 +vpunpckhqdq %xmm3,%xmm4,%xmm5 +vpunpcklqdq %xmm3,%xmm4,%xmm3 +vpmuludq %xmm3,%xmm10,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vpmuludq %xmm5,%xmm10,%xmm4 +vmovdqa %xmm5,224(%rsp) +vpaddq %xmm5,%xmm5,%xmm5 +vpmuludq %xmm3,%xmm12,%xmm14 +vmovdqa %xmm3,240(%rsp) +vpaddq %xmm14,%xmm4,%xmm4 +vpaddq %xmm7,%xmm6,%xmm3 +vpaddq subc2(%rip),%xmm6,%xmm6 +vpsubq %xmm7,%xmm6,%xmm6 +vpunpckhqdq %xmm3,%xmm6,%xmm7 +vpunpcklqdq %xmm3,%xmm6,%xmm3 +vpmuludq %xmm3,%xmm10,%xmm6 +vpmuludq %xmm5,%xmm12,%xmm14 +vmovdqa %xmm5,256(%rsp) +vpmuludq v19_19(%rip),%xmm5,%xmm5 +vmovdqa %xmm5,272(%rsp) +vpaddq %xmm14,%xmm6,%xmm6 +vpmuludq %xmm7,%xmm10,%xmm5 +vmovdqa %xmm7,288(%rsp) +vpaddq %xmm7,%xmm7,%xmm7 +vpmuludq %xmm3,%xmm12,%xmm14 +vmovdqa %xmm3,304(%rsp) +vpaddq %xmm14,%xmm5,%xmm5 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vmovdqa %xmm3,320(%rsp) +vpaddq %xmm9,%xmm8,%xmm3 +vpaddq subc2(%rip),%xmm8,%xmm8 +vpsubq %xmm9,%xmm8,%xmm8 +vpunpckhqdq %xmm3,%xmm8,%xmm9 +vpunpcklqdq %xmm3,%xmm8,%xmm3 +vmovdqa %xmm3,336(%rsp) +vpmuludq %xmm7,%xmm12,%xmm8 +vmovdqa %xmm7,352(%rsp) +vpmuludq v19_19(%rip),%xmm7,%xmm7 +vmovdqa %xmm7,368(%rsp) +vpmuludq %xmm3,%xmm10,%xmm7 +vpaddq %xmm7,%xmm8,%xmm8 +vpmuludq %xmm9,%xmm10,%xmm7 +vmovdqa %xmm9,384(%rsp) +vpaddq %xmm9,%xmm9,%xmm9 +vpmuludq %xmm3,%xmm12,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vmovdqa %xmm3,400(%rsp) +vpmuludq v19_19(%rip),%xmm12,%xmm12 +vpmuludq %xmm9,%xmm12,%xmm3 +vmovdqa %xmm9,416(%rsp) +vpaddq %xmm3,%xmm11,%xmm11 +vmovdqa 0(%rsp),%xmm3 +vmovdqa 16(%rsp),%xmm9 +vpaddq subc2(%rip),%xmm3,%xmm10 +vpsubq %xmm9,%xmm10,%xmm10 +vpaddq %xmm9,%xmm3,%xmm3 +vpunpckhqdq %xmm10,%xmm3,%xmm9 +vpunpcklqdq %xmm10,%xmm3,%xmm3 +vpmuludq 144(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm0,%xmm0 +vpmuludq 128(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm2,%xmm2 +vpmuludq 192(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpmuludq 176(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm4,%xmm4 +vpmuludq 240(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm6,%xmm6 +vpmuludq 224(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm5,%xmm5 +vpmuludq 304(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpmuludq 288(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vpmuludq 336(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm11,%xmm11 +vpmuludq 384(%rsp),%xmm3,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 144(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpmuludq 160(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm1,%xmm1 +vpmuludq 192(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 208(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 240(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 256(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm8,%xmm8 +vpmuludq 304(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm9,%xmm9 +vpmuludq 352(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 336(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 416(%rsp),%xmm9,%xmm9 +vpaddq %xmm9,%xmm0,%xmm0 +vmovdqa 32(%rsp),%xmm3 +vmovdqa 80(%rsp),%xmm9 +vpaddq subc2(%rip),%xmm3,%xmm10 +vpsubq %xmm9,%xmm10,%xmm10 +vpaddq %xmm9,%xmm3,%xmm3 +vpunpckhqdq %xmm10,%xmm3,%xmm9 +vpunpcklqdq %xmm10,%xmm3,%xmm3 +vpmuludq 144(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpmuludq 128(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm4,%xmm4 +vpmuludq 192(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm6,%xmm6 +vpmuludq 176(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm5,%xmm5 +vpmuludq 240(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpmuludq 224(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vpmuludq 304(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm11,%xmm11 +vpmuludq 288(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm13,%xmm13 +vpmuludq 336(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm0,%xmm0 +vpmuludq 384(%rsp),%xmm3,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpmuludq 144(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 160(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 192(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 208(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm8,%xmm8 +vpmuludq 240(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm9,%xmm9 +vpmuludq 256(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 304(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 352(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm0,%xmm0 +vpmuludq 336(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpmuludq 416(%rsp),%xmm9,%xmm9 +vpaddq %xmm9,%xmm1,%xmm1 +vmovdqa 48(%rsp),%xmm3 +vmovdqa 96(%rsp),%xmm9 +vpaddq subc2(%rip),%xmm3,%xmm10 +vpsubq %xmm9,%xmm10,%xmm10 +vpaddq %xmm9,%xmm3,%xmm3 +vpunpckhqdq %xmm10,%xmm3,%xmm9 +vpunpcklqdq %xmm10,%xmm3,%xmm3 +vpmuludq 144(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm6,%xmm6 +vpmuludq 128(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm5,%xmm5 +vpmuludq 192(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpmuludq 176(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vpmuludq 240(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm11,%xmm11 +vpmuludq 224(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm13,%xmm13 +vpmuludq 304(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm0,%xmm0 +vpmuludq 288(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm2,%xmm2 +vpmuludq 336(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpmuludq 384(%rsp),%xmm3,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 144(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 160(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm8,%xmm8 +vpmuludq 192(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm9,%xmm9 +vpmuludq 208(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 240(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 256(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm0,%xmm0 +vpmuludq 304(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpmuludq 352(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm1,%xmm1 +vpmuludq 336(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 416(%rsp),%xmm9,%xmm9 +vpaddq %xmm9,%xmm6,%xmm6 +vmovdqa 64(%rsp),%xmm3 +vmovdqa 112(%rsp),%xmm9 +vpaddq subc2(%rip),%xmm3,%xmm10 +vpsubq %xmm9,%xmm10,%xmm10 +vpaddq %xmm9,%xmm3,%xmm3 +vpunpckhqdq %xmm10,%xmm3,%xmm9 +vpunpcklqdq %xmm10,%xmm3,%xmm3 +vpmuludq 144(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpmuludq 128(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm3,%xmm3 +vpmuludq 192(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm11,%xmm11 +vpmuludq 176(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm13,%xmm13 +vpmuludq 240(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm0,%xmm0 +vpmuludq 224(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm2,%xmm2 +vpmuludq 304(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpmuludq 288(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm4,%xmm4 +vpmuludq 336(%rsp),%xmm3,%xmm10 +vpaddq %xmm10,%xmm6,%xmm6 +vpmuludq 384(%rsp),%xmm3,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 144(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq v19_19(%rip),%xmm9,%xmm9 +vpmuludq 160(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 192(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 208(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm0,%xmm0 +vpmuludq 240(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpmuludq 256(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm1,%xmm1 +vpmuludq 304(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpmuludq 352(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 336(%rsp),%xmm9,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 416(%rsp),%xmm9,%xmm9 +vpaddq %xmm9,%xmm8,%xmm8 +vpsrlq $25,%xmm4,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpand m25(%rip),%xmm4,%xmm4 +vpsrlq $26,%xmm11,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpand m26(%rip),%xmm11,%xmm11 +vpsrlq $26,%xmm6,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpand m26(%rip),%xmm6,%xmm6 +vpsrlq $25,%xmm13,%xmm3 +vpaddq %xmm3,%xmm0,%xmm0 +vpand m25(%rip),%xmm13,%xmm13 +vpsrlq $25,%xmm5,%xmm3 +vpaddq %xmm3,%xmm8,%xmm8 +vpand m25(%rip),%xmm5,%xmm5 +vpsrlq $26,%xmm0,%xmm3 +vpaddq %xmm3,%xmm2,%xmm2 +vpand m26(%rip),%xmm0,%xmm0 +vpsrlq $26,%xmm8,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpand m26(%rip),%xmm8,%xmm8 +vpsrlq $25,%xmm2,%xmm3 +vpaddq %xmm3,%xmm1,%xmm1 +vpand m25(%rip),%xmm2,%xmm2 +vpsrlq $25,%xmm7,%xmm3 +vpsllq $4,%xmm3,%xmm9 +vpaddq %xmm3,%xmm11,%xmm11 +vpsllq $1,%xmm3,%xmm3 +vpaddq %xmm3,%xmm9,%xmm9 +vpaddq %xmm9,%xmm11,%xmm11 +vpand m25(%rip),%xmm7,%xmm7 +vpsrlq $26,%xmm1,%xmm3 +vpaddq %xmm3,%xmm4,%xmm4 +vpand m26(%rip),%xmm1,%xmm1 +vpsrlq $26,%xmm11,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpand m26(%rip),%xmm11,%xmm11 +vpsrlq $25,%xmm4,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpand m25(%rip),%xmm4,%xmm4 +vpunpcklqdq %xmm13,%xmm11,%xmm3 +vpunpckhqdq %xmm13,%xmm11,%xmm9 +vpaddq subc0(%rip),%xmm9,%xmm10 +vpsubq %xmm3,%xmm10,%xmm10 +vpaddq %xmm9,%xmm3,%xmm3 +vpunpckhqdq %xmm3,%xmm10,%xmm9 +vpunpcklqdq %xmm3,%xmm10,%xmm10 +vpmuludq %xmm10,%xmm10,%xmm3 +vpaddq %xmm10,%xmm10,%xmm10 +vpmuludq %xmm9,%xmm10,%xmm11 +vpunpcklqdq %xmm2,%xmm0,%xmm12 +vpunpckhqdq %xmm2,%xmm0,%xmm0 +vpaddq subc2(%rip),%xmm0,%xmm2 +vpsubq %xmm12,%xmm2,%xmm2 +vpaddq %xmm0,%xmm12,%xmm12 +vpunpckhqdq %xmm12,%xmm2,%xmm0 +vpunpcklqdq %xmm12,%xmm2,%xmm2 +vpmuludq %xmm2,%xmm10,%xmm12 +vpaddq %xmm9,%xmm9,%xmm13 +vpmuludq %xmm13,%xmm9,%xmm9 +vpaddq %xmm9,%xmm12,%xmm12 +vpmuludq %xmm0,%xmm10,%xmm9 +vpmuludq %xmm2,%xmm13,%xmm14 +vpaddq %xmm14,%xmm9,%xmm9 +vpunpcklqdq %xmm4,%xmm1,%xmm14 +vpunpckhqdq %xmm4,%xmm1,%xmm1 +vpaddq subc2(%rip),%xmm1,%xmm4 +vpsubq %xmm14,%xmm4,%xmm4 +vpaddq %xmm1,%xmm14,%xmm14 +vpunpckhqdq %xmm14,%xmm4,%xmm1 +vpunpcklqdq %xmm14,%xmm4,%xmm4 +vmovdqa %xmm1,0(%rsp) +vpaddq %xmm1,%xmm1,%xmm1 +vmovdqa %xmm1,16(%rsp) +vpmuludq v19_19(%rip),%xmm1,%xmm1 +vmovdqa %xmm1,32(%rsp) +vpmuludq %xmm4,%xmm10,%xmm1 +vpmuludq %xmm2,%xmm2,%xmm14 +vpaddq %xmm14,%xmm1,%xmm1 +vpmuludq 0(%rsp),%xmm10,%xmm14 +vpmuludq %xmm4,%xmm13,%xmm15 +vpaddq %xmm15,%xmm14,%xmm14 +vpunpcklqdq %xmm5,%xmm6,%xmm15 +vpunpckhqdq %xmm5,%xmm6,%xmm5 +vpaddq subc2(%rip),%xmm5,%xmm6 +vpsubq %xmm15,%xmm6,%xmm6 +vpaddq %xmm5,%xmm15,%xmm15 +vpunpckhqdq %xmm15,%xmm6,%xmm5 +vpunpcklqdq %xmm15,%xmm6,%xmm6 +vmovdqa %xmm6,48(%rsp) +vpmuludq v19_19(%rip),%xmm6,%xmm6 +vmovdqa %xmm6,64(%rsp) +vmovdqa %xmm5,80(%rsp) +vpmuludq v38_38(%rip),%xmm5,%xmm5 +vmovdqa %xmm5,96(%rsp) +vpmuludq 48(%rsp),%xmm10,%xmm5 +vpaddq %xmm0,%xmm0,%xmm6 +vpmuludq %xmm6,%xmm0,%xmm0 +vpaddq %xmm0,%xmm5,%xmm5 +vpmuludq 80(%rsp),%xmm10,%xmm0 +vpmuludq %xmm4,%xmm6,%xmm15 +vpaddq %xmm15,%xmm0,%xmm0 +vpmuludq %xmm6,%xmm13,%xmm15 +vpaddq %xmm15,%xmm1,%xmm1 +vpmuludq %xmm6,%xmm2,%xmm15 +vpaddq %xmm15,%xmm14,%xmm14 +vpunpcklqdq %xmm7,%xmm8,%xmm15 +vpunpckhqdq %xmm7,%xmm8,%xmm7 +vpaddq subc2(%rip),%xmm7,%xmm8 +vpsubq %xmm15,%xmm8,%xmm8 +vpaddq %xmm7,%xmm15,%xmm15 +vpunpckhqdq %xmm15,%xmm8,%xmm7 +vpunpcklqdq %xmm15,%xmm8,%xmm8 +vmovdqa %xmm8,112(%rsp) +vpmuludq v19_19(%rip),%xmm8,%xmm8 +vmovdqa %xmm8,160(%rsp) +vpmuludq 112(%rsp),%xmm10,%xmm8 +vpmuludq %xmm7,%xmm10,%xmm10 +vpmuludq v38_38(%rip),%xmm7,%xmm15 +vpmuludq %xmm15,%xmm7,%xmm7 +vpaddq %xmm7,%xmm8,%xmm8 +vpmuludq %xmm15,%xmm13,%xmm7 +vpaddq %xmm7,%xmm3,%xmm3 +vpmuludq %xmm15,%xmm2,%xmm7 +vpaddq %xmm7,%xmm11,%xmm11 +vpmuludq 80(%rsp),%xmm13,%xmm7 +vpaddq %xmm7,%xmm7,%xmm7 +vpaddq %xmm7,%xmm8,%xmm8 +vpmuludq 16(%rsp),%xmm13,%xmm7 +vpaddq %xmm7,%xmm5,%xmm5 +vpmuludq 48(%rsp),%xmm13,%xmm7 +vpaddq %xmm7,%xmm0,%xmm0 +vpmuludq 112(%rsp),%xmm13,%xmm7 +vpaddq %xmm7,%xmm10,%xmm10 +vpmuludq %xmm15,%xmm6,%xmm7 +vpaddq %xmm7,%xmm12,%xmm12 +vpmuludq %xmm15,%xmm4,%xmm7 +vpaddq %xmm7,%xmm9,%xmm9 +vpaddq %xmm2,%xmm2,%xmm2 +vpmuludq %xmm4,%xmm2,%xmm7 +vpaddq %xmm7,%xmm5,%xmm5 +vpmuludq 160(%rsp),%xmm2,%xmm7 +vpaddq %xmm7,%xmm3,%xmm3 +vpmuludq 160(%rsp),%xmm6,%xmm7 +vpaddq %xmm7,%xmm11,%xmm11 +vpmuludq 0(%rsp),%xmm2,%xmm7 +vpaddq %xmm7,%xmm0,%xmm0 +vpmuludq 48(%rsp),%xmm2,%xmm7 +vpaddq %xmm7,%xmm8,%xmm8 +vpmuludq 80(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 96(%rsp),%xmm4,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpmuludq %xmm4,%xmm4,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpaddq %xmm4,%xmm4,%xmm2 +vpmuludq 160(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm12,%xmm12 +vpmuludq 16(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vpmuludq 48(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm14,%xmm14 +vpmuludq 96(%rsp),%xmm6,%xmm4 +vpaddq %xmm4,%xmm3,%xmm3 +vmovdqa 16(%rsp),%xmm4 +vpmuludq 160(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm9,%xmm9 +vpmuludq 16(%rsp),%xmm6,%xmm4 +vpaddq %xmm4,%xmm8,%xmm8 +vpmuludq 48(%rsp),%xmm6,%xmm4 +vpaddq %xmm4,%xmm10,%xmm10 +vpmuludq 80(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm4,%xmm4 +vpaddq %xmm4,%xmm5,%xmm5 +vpmuludq 112(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm0,%xmm0 +vmovdqa 48(%rsp),%xmm4 +vpaddq %xmm4,%xmm4,%xmm4 +vpmuludq 160(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vmovdqa 80(%rsp),%xmm4 +vpaddq %xmm4,%xmm4,%xmm4 +vpmuludq 160(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm14,%xmm14 +vpmuludq 64(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm3,%xmm3 +vmovdqa 16(%rsp),%xmm4 +vpmuludq 64(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm11,%xmm11 +vmovdqa 16(%rsp),%xmm4 +vpmuludq 96(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm12,%xmm12 +vmovdqa 48(%rsp),%xmm4 +vpmuludq 96(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm9,%xmm9 +vpmuludq 0(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vmovdqa 32(%rsp),%xmm2 +vpmuludq 0(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm3,%xmm3 +vmovdqa 64(%rsp),%xmm2 +vpmuludq 48(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vmovdqa 96(%rsp),%xmm2 +vpmuludq 80(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm1,%xmm1 +vmovdqa 160(%rsp),%xmm2 +vpmuludq 112(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpsrlq $26,%xmm3,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpand m26(%rip),%xmm3,%xmm3 +vpsrlq $25,%xmm14,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpand m25(%rip),%xmm14,%xmm14 +vpsrlq $25,%xmm11,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpand m25(%rip),%xmm11,%xmm11 +vpsrlq $26,%xmm5,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vpand m26(%rip),%xmm5,%xmm5 +vpsrlq $26,%xmm12,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpand m26(%rip),%xmm12,%xmm12 +vpsrlq $25,%xmm0,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpand m25(%rip),%xmm0,%xmm0 +vpsrlq $25,%xmm9,%xmm2 +vpaddq %xmm2,%xmm1,%xmm1 +vpand m25(%rip),%xmm9,%xmm9 +vpsrlq $26,%xmm8,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpand m26(%rip),%xmm8,%xmm8 +vpsrlq $26,%xmm1,%xmm2 +vpaddq %xmm2,%xmm14,%xmm14 +vpand m26(%rip),%xmm1,%xmm1 +vpsrlq $25,%xmm10,%xmm2 +vpsllq $4,%xmm2,%xmm4 +vpaddq %xmm2,%xmm3,%xmm3 +vpsllq $1,%xmm2,%xmm2 +vpaddq %xmm2,%xmm4,%xmm4 +vpaddq %xmm4,%xmm3,%xmm3 +vpand m25(%rip),%xmm10,%xmm10 +vpsrlq $25,%xmm14,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpand m25(%rip),%xmm14,%xmm14 +vpsrlq $26,%xmm3,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpand m26(%rip),%xmm3,%xmm3 +vpunpckhqdq %xmm11,%xmm3,%xmm2 +vmovdqa %xmm2,0(%rsp) +vpunpcklqdq %xmm11,%xmm3,%xmm2 +vpmuludq v9_9(%rip),%xmm2,%xmm2 +vmovdqa %xmm2,80(%rsp) +vpunpckhqdq %xmm9,%xmm12,%xmm2 +vmovdqa %xmm2,16(%rsp) +vpunpcklqdq %xmm9,%xmm12,%xmm2 +vpmuludq v9_9(%rip),%xmm2,%xmm2 +vmovdqa %xmm2,96(%rsp) +vpunpckhqdq %xmm14,%xmm1,%xmm2 +vmovdqa %xmm2,32(%rsp) +vpunpcklqdq %xmm14,%xmm1,%xmm1 +vpmuludq v9_9(%rip),%xmm1,%xmm1 +vmovdqa %xmm1,112(%rsp) +vpunpckhqdq %xmm0,%xmm5,%xmm1 +vmovdqa %xmm1,48(%rsp) +vpunpcklqdq %xmm0,%xmm5,%xmm0 +vpmuludq v9_9(%rip),%xmm0,%xmm0 +vmovdqa %xmm0,160(%rsp) +vpunpckhqdq %xmm10,%xmm8,%xmm0 +vmovdqa %xmm0,64(%rsp) +vpunpcklqdq %xmm10,%xmm8,%xmm0 +vpmuludq v9_9(%rip),%xmm0,%xmm0 +vmovdqa %xmm0,208(%rsp) +vmovdqa 144(%rsp),%xmm0 +vpmuludq %xmm0,%xmm0,%xmm1 +vpaddq %xmm0,%xmm0,%xmm0 +vmovdqa 128(%rsp),%xmm2 +vpmuludq %xmm2,%xmm0,%xmm3 +vmovdqa 192(%rsp),%xmm4 +vpmuludq %xmm4,%xmm0,%xmm5 +vmovdqa 176(%rsp),%xmm6 +vpmuludq %xmm6,%xmm0,%xmm7 +vmovdqa 240(%rsp),%xmm8 +vpmuludq %xmm8,%xmm0,%xmm9 +vpmuludq 224(%rsp),%xmm0,%xmm10 +vpmuludq 304(%rsp),%xmm0,%xmm11 +vpmuludq 288(%rsp),%xmm0,%xmm12 +vpmuludq 336(%rsp),%xmm0,%xmm13 +vmovdqa 384(%rsp),%xmm14 +vpmuludq %xmm14,%xmm0,%xmm0 +vpmuludq v38_38(%rip),%xmm14,%xmm15 +vpmuludq %xmm15,%xmm14,%xmm14 +vpaddq %xmm14,%xmm13,%xmm13 +vpaddq %xmm6,%xmm6,%xmm14 +vpmuludq %xmm14,%xmm6,%xmm6 +vpaddq %xmm6,%xmm11,%xmm11 +vpaddq %xmm2,%xmm2,%xmm6 +vpmuludq %xmm6,%xmm2,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpmuludq %xmm15,%xmm6,%xmm2 +vpaddq %xmm2,%xmm1,%xmm1 +vpmuludq %xmm15,%xmm4,%xmm2 +vpaddq %xmm2,%xmm3,%xmm3 +vpmuludq 256(%rsp),%xmm6,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpmuludq 304(%rsp),%xmm6,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 352(%rsp),%xmm6,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpmuludq 336(%rsp),%xmm6,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vpmuludq %xmm4,%xmm6,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpmuludq %xmm14,%xmm6,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpmuludq %xmm8,%xmm6,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq %xmm15,%xmm14,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpmuludq %xmm15,%xmm8,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpmuludq %xmm4,%xmm4,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpmuludq %xmm14,%xmm4,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpaddq %xmm4,%xmm4,%xmm2 +vpmuludq %xmm8,%xmm2,%xmm4 +vpaddq %xmm4,%xmm11,%xmm11 +vpmuludq 400(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vpmuludq 400(%rsp),%xmm14,%xmm4 +vpaddq %xmm4,%xmm3,%xmm3 +vpmuludq 224(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm12,%xmm12 +vpmuludq 304(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm13,%xmm13 +vpmuludq 288(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vpmuludq 368(%rsp),%xmm8,%xmm2 +vpaddq %xmm2,%xmm3,%xmm3 +vpmuludq %xmm8,%xmm14,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq %xmm8,%xmm8,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpaddq %xmm8,%xmm8,%xmm2 +vpmuludq 400(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm5,%xmm5 +vpmuludq 256(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm9,%xmm9 +vpmuludq 304(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm10,%xmm10 +vpmuludq 368(%rsp),%xmm14,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vmovdqa 256(%rsp),%xmm4 +vpmuludq 400(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm7,%xmm7 +vpmuludq 256(%rsp),%xmm14,%xmm4 +vpaddq %xmm4,%xmm13,%xmm13 +vpmuludq 304(%rsp),%xmm14,%xmm4 +vpaddq %xmm4,%xmm0,%xmm0 +vpmuludq 352(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm11,%xmm11 +vpmuludq 336(%rsp),%xmm15,%xmm4 +vpaddq %xmm4,%xmm12,%xmm12 +vmovdqa 304(%rsp),%xmm4 +vpaddq %xmm4,%xmm4,%xmm4 +vpmuludq 400(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm9,%xmm9 +vpmuludq 320(%rsp),%xmm2,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vmovdqa 256(%rsp),%xmm4 +vpmuludq 320(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm3,%xmm3 +vmovdqa 256(%rsp),%xmm4 +vpmuludq 368(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm5,%xmm5 +vmovdqa 304(%rsp),%xmm4 +vpmuludq 368(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm7,%xmm7 +vmovdqa 352(%rsp),%xmm4 +vpmuludq 400(%rsp),%xmm4,%xmm4 +vpaddq %xmm4,%xmm10,%xmm10 +vpmuludq 224(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vmovdqa 272(%rsp),%xmm2 +vpmuludq 224(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm1,%xmm1 +vmovdqa 320(%rsp),%xmm2 +vpmuludq 304(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vmovdqa 368(%rsp),%xmm2 +vpmuludq 288(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vmovdqa 400(%rsp),%xmm2 +vpmuludq 336(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpsrlq $26,%xmm1,%xmm2 +vpaddq %xmm2,%xmm3,%xmm3 +vpand m26(%rip),%xmm1,%xmm1 +vpsrlq $25,%xmm10,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpand m25(%rip),%xmm10,%xmm10 +vpsrlq $25,%xmm3,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpand m25(%rip),%xmm3,%xmm3 +vpsrlq $26,%xmm11,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpand m26(%rip),%xmm11,%xmm11 +vpsrlq $26,%xmm5,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpand m26(%rip),%xmm5,%xmm5 +vpsrlq $25,%xmm12,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpand m25(%rip),%xmm12,%xmm12 +vpsrlq $25,%xmm7,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpand m25(%rip),%xmm7,%xmm7 +vpsrlq $26,%xmm13,%xmm2 +vpaddq %xmm2,%xmm0,%xmm0 +vpand m26(%rip),%xmm13,%xmm13 +vpsrlq $26,%xmm9,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpand m26(%rip),%xmm9,%xmm9 +vpsrlq $25,%xmm0,%xmm2 +vpsllq $4,%xmm2,%xmm4 +vpaddq %xmm2,%xmm1,%xmm1 +vpsllq $1,%xmm2,%xmm2 +vpaddq %xmm2,%xmm4,%xmm4 +vpaddq %xmm4,%xmm1,%xmm1 +vpand m25(%rip),%xmm0,%xmm0 +vpsrlq $25,%xmm10,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpand m25(%rip),%xmm10,%xmm10 +vpsrlq $26,%xmm1,%xmm2 +vpaddq %xmm2,%xmm3,%xmm3 +vpand m26(%rip),%xmm1,%xmm1 +vpunpckhqdq %xmm3,%xmm1,%xmm2 +vpunpcklqdq %xmm3,%xmm1,%xmm1 +vmovdqa %xmm1,176(%rsp) +vpaddq subc0(%rip),%xmm2,%xmm3 +vpsubq %xmm1,%xmm3,%xmm3 +vpunpckhqdq %xmm3,%xmm2,%xmm1 +vpunpcklqdq %xmm3,%xmm2,%xmm2 +vmovdqa %xmm2,192(%rsp) +vmovdqa %xmm1,224(%rsp) +vpsllq $1,%xmm1,%xmm1 +vmovdqa %xmm1,240(%rsp) +vpmuludq v121666_121666(%rip),%xmm3,%xmm3 +vmovdqa 80(%rsp),%xmm1 +vpunpcklqdq %xmm1,%xmm3,%xmm2 +vpunpckhqdq %xmm1,%xmm3,%xmm1 +vpunpckhqdq %xmm7,%xmm5,%xmm3 +vpunpcklqdq %xmm7,%xmm5,%xmm4 +vmovdqa %xmm4,256(%rsp) +vpaddq subc2(%rip),%xmm3,%xmm5 +vpsubq %xmm4,%xmm5,%xmm5 +vpunpckhqdq %xmm5,%xmm3,%xmm4 +vpunpcklqdq %xmm5,%xmm3,%xmm3 +vmovdqa %xmm3,272(%rsp) +vmovdqa %xmm4,288(%rsp) +vpsllq $1,%xmm4,%xmm4 +vmovdqa %xmm4,304(%rsp) +vpmuludq v121666_121666(%rip),%xmm5,%xmm5 +vmovdqa 96(%rsp),%xmm3 +vpunpcklqdq %xmm3,%xmm5,%xmm4 +vpunpckhqdq %xmm3,%xmm5,%xmm3 +vpunpckhqdq %xmm10,%xmm9,%xmm5 +vpunpcklqdq %xmm10,%xmm9,%xmm6 +vmovdqa %xmm6,320(%rsp) +vpaddq subc2(%rip),%xmm5,%xmm7 +vpsubq %xmm6,%xmm7,%xmm7 +vpunpckhqdq %xmm7,%xmm5,%xmm6 +vpunpcklqdq %xmm7,%xmm5,%xmm5 +vmovdqa %xmm5,336(%rsp) +vmovdqa %xmm6,352(%rsp) +vpsllq $1,%xmm6,%xmm6 +vmovdqa %xmm6,368(%rsp) +vpmuludq v121666_121666(%rip),%xmm7,%xmm7 +vmovdqa 112(%rsp),%xmm5 +vpunpcklqdq %xmm5,%xmm7,%xmm6 +vpunpckhqdq %xmm5,%xmm7,%xmm5 +vpunpckhqdq %xmm12,%xmm11,%xmm7 +vpunpcklqdq %xmm12,%xmm11,%xmm8 +vmovdqa %xmm8,384(%rsp) +vpaddq subc2(%rip),%xmm7,%xmm9 +vpsubq %xmm8,%xmm9,%xmm9 +vpunpckhqdq %xmm9,%xmm7,%xmm8 +vpunpcklqdq %xmm9,%xmm7,%xmm7 +vmovdqa %xmm7,400(%rsp) +vmovdqa %xmm8,416(%rsp) +vpsllq $1,%xmm8,%xmm8 +vmovdqa %xmm8,432(%rsp) +vpmuludq v121666_121666(%rip),%xmm9,%xmm9 +vmovdqa 160(%rsp),%xmm7 +vpunpcklqdq %xmm7,%xmm9,%xmm8 +vpunpckhqdq %xmm7,%xmm9,%xmm7 +vpunpckhqdq %xmm0,%xmm13,%xmm9 +vpunpcklqdq %xmm0,%xmm13,%xmm0 +vmovdqa %xmm0,160(%rsp) +vpaddq subc2(%rip),%xmm9,%xmm10 +vpsubq %xmm0,%xmm10,%xmm10 +vpunpckhqdq %xmm10,%xmm9,%xmm0 +vpunpcklqdq %xmm10,%xmm9,%xmm9 +vmovdqa %xmm9,448(%rsp) +vmovdqa %xmm0,464(%rsp) +vpsllq $1,%xmm0,%xmm0 +vmovdqa %xmm0,480(%rsp) +vpmuludq v121666_121666(%rip),%xmm10,%xmm10 +vmovdqa 208(%rsp),%xmm0 +vpunpcklqdq %xmm0,%xmm10,%xmm9 +vpunpckhqdq %xmm0,%xmm10,%xmm0 +vpsrlq $26,%xmm2,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpand m26(%rip),%xmm2,%xmm2 +vpsrlq $25,%xmm5,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpand m25(%rip),%xmm5,%xmm5 +vpsrlq $25,%xmm1,%xmm10 +vpaddq %xmm10,%xmm4,%xmm4 +vpand m25(%rip),%xmm1,%xmm1 +vpsrlq $26,%xmm8,%xmm10 +vpaddq %xmm10,%xmm7,%xmm7 +vpand m26(%rip),%xmm8,%xmm8 +vpsrlq $26,%xmm4,%xmm10 +vpaddq %xmm10,%xmm3,%xmm3 +vpand m26(%rip),%xmm4,%xmm4 +vpsrlq $25,%xmm7,%xmm10 +vpaddq %xmm10,%xmm9,%xmm9 +vpand m25(%rip),%xmm7,%xmm7 +vpsrlq $25,%xmm3,%xmm10 +vpaddq %xmm10,%xmm6,%xmm6 +vpand m25(%rip),%xmm3,%xmm3 +vpsrlq $26,%xmm9,%xmm10 +vpaddq %xmm10,%xmm0,%xmm0 +vpand m26(%rip),%xmm9,%xmm9 +vpsrlq $26,%xmm6,%xmm10 +vpaddq %xmm10,%xmm5,%xmm5 +vpand m26(%rip),%xmm6,%xmm6 +vpsrlq $25,%xmm0,%xmm10 +vpsllq $4,%xmm10,%xmm11 +vpaddq %xmm10,%xmm2,%xmm2 +vpsllq $1,%xmm10,%xmm10 +vpaddq %xmm10,%xmm11,%xmm11 +vpaddq %xmm11,%xmm2,%xmm2 +vpand m25(%rip),%xmm0,%xmm0 +vpsrlq $25,%xmm5,%xmm10 +vpaddq %xmm10,%xmm8,%xmm8 +vpand m25(%rip),%xmm5,%xmm5 +vpsrlq $26,%xmm2,%xmm10 +vpaddq %xmm10,%xmm1,%xmm1 +vpand m26(%rip),%xmm2,%xmm2 +vpunpckhqdq %xmm1,%xmm2,%xmm10 +vmovdqa %xmm10,80(%rsp) +vpunpcklqdq %xmm1,%xmm2,%xmm1 +vpunpckhqdq %xmm3,%xmm4,%xmm2 +vmovdqa %xmm2,96(%rsp) +vpunpcklqdq %xmm3,%xmm4,%xmm2 +vpunpckhqdq %xmm5,%xmm6,%xmm3 +vmovdqa %xmm3,112(%rsp) +vpunpcklqdq %xmm5,%xmm6,%xmm3 +vpunpckhqdq %xmm7,%xmm8,%xmm4 +vmovdqa %xmm4,128(%rsp) +vpunpcklqdq %xmm7,%xmm8,%xmm4 +vpunpckhqdq %xmm0,%xmm9,%xmm5 +vmovdqa %xmm5,144(%rsp) +vpunpcklqdq %xmm0,%xmm9,%xmm0 +vmovdqa 176(%rsp),%xmm5 +vpaddq %xmm5,%xmm1,%xmm1 +vpunpcklqdq %xmm1,%xmm5,%xmm6 +vpunpckhqdq %xmm1,%xmm5,%xmm1 +vpmuludq 224(%rsp),%xmm6,%xmm5 +vpmuludq 192(%rsp),%xmm1,%xmm7 +vpaddq %xmm7,%xmm5,%xmm5 +vpmuludq 272(%rsp),%xmm6,%xmm7 +vpmuludq 240(%rsp),%xmm1,%xmm8 +vpaddq %xmm8,%xmm7,%xmm7 +vpmuludq 288(%rsp),%xmm6,%xmm8 +vpmuludq 272(%rsp),%xmm1,%xmm9 +vpaddq %xmm9,%xmm8,%xmm8 +vpmuludq 336(%rsp),%xmm6,%xmm9 +vpmuludq 304(%rsp),%xmm1,%xmm10 +vpaddq %xmm10,%xmm9,%xmm9 +vpmuludq 352(%rsp),%xmm6,%xmm10 +vpmuludq 336(%rsp),%xmm1,%xmm11 +vpaddq %xmm11,%xmm10,%xmm10 +vpmuludq 400(%rsp),%xmm6,%xmm11 +vpmuludq 368(%rsp),%xmm1,%xmm12 +vpaddq %xmm12,%xmm11,%xmm11 +vpmuludq 416(%rsp),%xmm6,%xmm12 +vpmuludq 400(%rsp),%xmm1,%xmm13 +vpaddq %xmm13,%xmm12,%xmm12 +vpmuludq 448(%rsp),%xmm6,%xmm13 +vpmuludq 432(%rsp),%xmm1,%xmm14 +vpaddq %xmm14,%xmm13,%xmm13 +vpmuludq 464(%rsp),%xmm6,%xmm14 +vpmuludq 448(%rsp),%xmm1,%xmm15 +vpaddq %xmm15,%xmm14,%xmm14 +vpmuludq 192(%rsp),%xmm6,%xmm6 +vpmuludq v19_19(%rip),%xmm1,%xmm1 +vpmuludq 480(%rsp),%xmm1,%xmm1 +vpaddq %xmm1,%xmm6,%xmm6 +vmovdqa 256(%rsp),%xmm1 +vpaddq %xmm1,%xmm2,%xmm2 +vpunpcklqdq %xmm2,%xmm1,%xmm15 +vpunpckhqdq %xmm2,%xmm1,%xmm1 +vpmuludq 192(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpmuludq 224(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpmuludq 272(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpmuludq 288(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 336(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpmuludq 352(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 400(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpmuludq 416(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm15,%xmm15 +vpmuludq 448(%rsp),%xmm15,%xmm2 +vpaddq %xmm2,%xmm6,%xmm6 +vpmuludq 464(%rsp),%xmm15,%xmm15 +vpaddq %xmm15,%xmm5,%xmm5 +vpmuludq 192(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpmuludq 240(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpmuludq 272(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 304(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpmuludq 336(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 368(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpmuludq 400(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm1,%xmm1 +vpmuludq 432(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm6,%xmm6 +vpmuludq 448(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpmuludq 480(%rsp),%xmm1,%xmm1 +vpaddq %xmm1,%xmm7,%xmm7 +vmovdqa 320(%rsp),%xmm1 +vpaddq %xmm1,%xmm3,%xmm3 +vpunpcklqdq %xmm3,%xmm1,%xmm2 +vpunpckhqdq %xmm3,%xmm1,%xmm1 +vpmuludq 192(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm9,%xmm9 +vpmuludq 224(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm10,%xmm10 +vpmuludq 272(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 288(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm12,%xmm12 +vpmuludq 336(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 352(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm2,%xmm2 +vpmuludq 400(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 416(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 448(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq 464(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpmuludq 192(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 240(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm11,%xmm11 +vpmuludq 272(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 304(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpmuludq 336(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm1,%xmm1 +vpmuludq 368(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm6,%xmm6 +vpmuludq 400(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpmuludq 432(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpmuludq 448(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpmuludq 480(%rsp),%xmm1,%xmm1 +vpaddq %xmm1,%xmm9,%xmm9 +vmovdqa 384(%rsp),%xmm1 +vpaddq %xmm1,%xmm4,%xmm4 +vpunpcklqdq %xmm4,%xmm1,%xmm2 +vpunpckhqdq %xmm4,%xmm1,%xmm1 +vpmuludq 192(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm11,%xmm11 +vpmuludq 224(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm12,%xmm12 +vpmuludq 272(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm13,%xmm13 +vpmuludq 288(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm2,%xmm2 +vpmuludq 336(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm6,%xmm6 +vpmuludq 352(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm5,%xmm5 +vpmuludq 400(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm7,%xmm7 +vpmuludq 416(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm8,%xmm8 +vpmuludq 448(%rsp),%xmm2,%xmm3 +vpaddq %xmm3,%xmm9,%xmm9 +vpmuludq 464(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 192(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 240(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm13,%xmm13 +vpmuludq 272(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm1,%xmm1 +vpmuludq 304(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm6,%xmm6 +vpmuludq 336(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm5,%xmm5 +vpmuludq 368(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm7,%xmm7 +vpmuludq 400(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm8,%xmm8 +vpmuludq 432(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm9,%xmm9 +vpmuludq 448(%rsp),%xmm1,%xmm2 +vpaddq %xmm2,%xmm10,%xmm10 +vpmuludq 480(%rsp),%xmm1,%xmm1 +vpaddq %xmm1,%xmm11,%xmm11 +vmovdqa 160(%rsp),%xmm1 +vpaddq %xmm1,%xmm0,%xmm0 +vpunpcklqdq %xmm0,%xmm1,%xmm2 +vpunpckhqdq %xmm0,%xmm1,%xmm0 +vpmuludq 192(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm13,%xmm13 +vpmuludq 224(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm2,%xmm2 +vpmuludq 272(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm6,%xmm6 +vpmuludq 288(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm5,%xmm5 +vpmuludq 336(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm7,%xmm7 +vpmuludq 352(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm8,%xmm8 +vpmuludq 400(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm9,%xmm9 +vpmuludq 416(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm10,%xmm10 +vpmuludq 448(%rsp),%xmm2,%xmm1 +vpaddq %xmm1,%xmm11,%xmm11 +vpmuludq 464(%rsp),%xmm2,%xmm2 +vpaddq %xmm2,%xmm12,%xmm12 +vpmuludq 192(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm14,%xmm14 +vpmuludq v19_19(%rip),%xmm0,%xmm0 +vpmuludq 240(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm6,%xmm6 +vpmuludq 272(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm5,%xmm5 +vpmuludq 304(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm7,%xmm7 +vpmuludq 336(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm8,%xmm8 +vpmuludq 368(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm9,%xmm9 +vpmuludq 400(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm10,%xmm10 +vpmuludq 432(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm11,%xmm11 +vpmuludq 448(%rsp),%xmm0,%xmm1 +vpaddq %xmm1,%xmm12,%xmm12 +vpmuludq 480(%rsp),%xmm0,%xmm0 +vpaddq %xmm0,%xmm13,%xmm13 +vpsrlq $26,%xmm6,%xmm0 +vpaddq %xmm0,%xmm5,%xmm5 +vpand m26(%rip),%xmm6,%xmm6 +vpsrlq $25,%xmm10,%xmm0 +vpaddq %xmm0,%xmm11,%xmm11 +vpand m25(%rip),%xmm10,%xmm10 +vpsrlq $25,%xmm5,%xmm0 +vpaddq %xmm0,%xmm7,%xmm7 +vpand m25(%rip),%xmm5,%xmm5 +vpsrlq $26,%xmm11,%xmm0 +vpaddq %xmm0,%xmm12,%xmm12 +vpand m26(%rip),%xmm11,%xmm11 +vpsrlq $26,%xmm7,%xmm0 +vpaddq %xmm0,%xmm8,%xmm8 +vpand m26(%rip),%xmm7,%xmm7 +vpsrlq $25,%xmm12,%xmm0 +vpaddq %xmm0,%xmm13,%xmm13 +vpand m25(%rip),%xmm12,%xmm12 +vpsrlq $25,%xmm8,%xmm0 +vpaddq %xmm0,%xmm9,%xmm9 +vpand m25(%rip),%xmm8,%xmm8 +vpsrlq $26,%xmm13,%xmm0 +vpaddq %xmm0,%xmm14,%xmm14 +vpand m26(%rip),%xmm13,%xmm13 +vpsrlq $26,%xmm9,%xmm0 +vpaddq %xmm0,%xmm10,%xmm10 +vpand m26(%rip),%xmm9,%xmm9 +vpsrlq $25,%xmm14,%xmm0 +vpsllq $4,%xmm0,%xmm1 +vpaddq %xmm0,%xmm6,%xmm6 +vpsllq $1,%xmm0,%xmm0 +vpaddq %xmm0,%xmm1,%xmm1 +vpaddq %xmm1,%xmm6,%xmm6 +vpand m25(%rip),%xmm14,%xmm14 +vpsrlq $25,%xmm10,%xmm0 +vpaddq %xmm0,%xmm11,%xmm11 +vpand m25(%rip),%xmm10,%xmm10 +vpsrlq $26,%xmm6,%xmm0 +vpaddq %xmm0,%xmm5,%xmm5 +vpand m26(%rip),%xmm6,%xmm6 +vpunpckhqdq %xmm5,%xmm6,%xmm1 +vpunpcklqdq %xmm5,%xmm6,%xmm0 +vpunpckhqdq %xmm8,%xmm7,%xmm3 +vpunpcklqdq %xmm8,%xmm7,%xmm2 +vpunpckhqdq %xmm10,%xmm9,%xmm5 +vpunpcklqdq %xmm10,%xmm9,%xmm4 +vpunpckhqdq %xmm12,%xmm11,%xmm7 +vpunpcklqdq %xmm12,%xmm11,%xmm6 +vpunpckhqdq %xmm14,%xmm13,%xmm9 +vpunpcklqdq %xmm14,%xmm13,%xmm8 +cmp $0,%rdx +jne ._ladder_base_loop +vmovdqu %xmm1,80(%rdi) +vmovdqu %xmm0,0(%rdi) +vmovdqu %xmm3,96(%rdi) +vmovdqu %xmm2,16(%rdi) +vmovdqu %xmm5,112(%rdi) +vmovdqu %xmm4,32(%rdi) +vmovdqu %xmm7,128(%rdi) +vmovdqu %xmm6,48(%rdi) +vmovdqu %xmm9,144(%rdi) +vmovdqu %xmm8,64(%rdi) +movq 1536(%rsp),%r11 +movq 1544(%rsp),%r12 +movq 1552(%rsp),%r13 +add %r11,%rsp +ret + +#endif + +.globl ladder_base_S_dummy diff --git a/sodium/sodium_sandy2x_ladder_base.h b/sodium/sodium_sandy2x_ladder_base.h new file mode 100644 index 00000000..3b881c85 --- /dev/null +++ b/sodium/sodium_sandy2x_ladder_base.h @@ -0,0 +1,18 @@ +#ifndef ladder_base_H +#define ladder_base_H + +#ifdef __cplusplus +extern "C" { +#endif + +#include "sodium_sandy2x_fe.h" +#include "sodium_sandy2x_ladder_base_namespace.h" + +extern void ladder_base(fe *, const unsigned char *); + +#ifdef __cplusplus +} +#endif + +#endif /* ifndef ladder_base_H */ + diff --git a/sodium/sodium_sandy2x_ladder_base_namespace.h b/sodium/sodium_sandy2x_ladder_base_namespace.h new file mode 100644 index 00000000..304546a1 --- /dev/null +++ b/sodium/sodium_sandy2x_ladder_base_namespace.h @@ -0,0 +1,8 @@ +#ifndef ladder_base_namespace_H +#define ladder_base_namespace_H + +#define ladder_base crypto_scalarmult_curve25519_sandy2x_ladder_base +#define _ladder_base _crypto_scalarmult_curve25519_sandy2x_ladder_base + +#endif /* ifndef ladder_base_namespace_H */ + diff --git a/sodium/sodium_sandy2x_ladder_namespace.h b/sodium/sodium_sandy2x_ladder_namespace.h new file mode 100644 index 00000000..6637074b --- /dev/null +++ b/sodium/sodium_sandy2x_ladder_namespace.h @@ -0,0 +1,8 @@ +#ifndef ladder_namespace_H +#define ladder_namespace_H + +#define ladder crypto_scalarmult_curve25519_sandy2x_ladder +#define _ladder _crypto_scalarmult_curve25519_sandy2x_ladder + +#endif /* ifndef ladder_namespace_H */ + diff --git a/sodium/sodium_scalarmult_curve25519.c b/sodium/sodium_scalarmult_curve25519.c new file mode 100644 index 00000000..fa158860 --- /dev/null +++ b/sodium/sodium_scalarmult_curve25519.c @@ -0,0 +1,68 @@ + +#include "sodium_crypto_scalarmult_curve25519.h" +#include "sodium_private_implementations.h" +#include "sodium_private_common.h" +#include "sodium_scalarmult_curve25519.h" +#include "sodium_runtime.h" +#include + +#ifdef HAVE_AVX_ASM +# include "sodium_sandy2x_curve25519.h" +#endif +#include "sodium_ref10_x25519.h" +static const crypto_scalarmult_curve25519_implementation *implementation = + &crypto_scalarmult_curve25519_ref10_implementation; + +int +crypto_scalarmult_curve25519(unsigned char *q, const unsigned char *n, + const unsigned char *p) +{ + size_t i; + volatile unsigned char d = 0; + + if (implementation->mult(q, n, p) != 0) { + return -1; /* LCOV_EXCL_LINE */ + } + for (i = 0; i < crypto_scalarmult_curve25519_BYTES; i++) { + d |= q[i]; + } + return -(1 & ((d - 1) >> 8)); +} + +int +crypto_scalarmult_curve25519_base(unsigned char *q, const unsigned char *n) +{ + return implementation->mult_base(q, n); +} + +size_t +crypto_scalarmult_curve25519_bytes(void) +{ + return crypto_scalarmult_curve25519_BYTES; +} + +size_t +crypto_scalarmult_curve25519_scalarbytes(void) +{ + return crypto_scalarmult_curve25519_SCALARBYTES; +} + +int +_crypto_scalarmult_curve25519_pick_best_implementation(void) +{ + implementation = &crypto_scalarmult_curve25519_ref10_implementation; + +#ifdef HAVE_AVX_ASM + if (sodium_runtime_has_avx()) { + #if NETCODE_CRYPTO_LOGS + printf( "curve25519 -> avx\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + implementation = &crypto_scalarmult_curve25519_sandy2x_implementation; + return 0; + } +#endif + #if NETCODE_CRYPTO_LOGS + printf( "curve25519 -> ref\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + return 0; +} diff --git a/sodium/sodium_scalarmult_curve25519.h b/sodium/sodium_scalarmult_curve25519.h new file mode 100644 index 00000000..66edbf6a --- /dev/null +++ b/sodium/sodium_scalarmult_curve25519.h @@ -0,0 +1,11 @@ + +#ifndef scalarmult_poly1305_H +#define scalarmult_poly1305_H + +typedef struct crypto_scalarmult_curve25519_implementation { + int (*mult)(unsigned char *q, const unsigned char *n, + const unsigned char *p); + int (*mult_base)(unsigned char *q, const unsigned char *n); +} crypto_scalarmult_curve25519_implementation; + +#endif diff --git a/sodium/sodium_secretbox_xchacha20poly1305.c b/sodium/sodium_secretbox_xchacha20poly1305.c new file mode 100644 index 00000000..35fd125b --- /dev/null +++ b/sodium/sodium_secretbox_xchacha20poly1305.c @@ -0,0 +1,177 @@ + +#include +#include +#include +#include +#include + +#include "sodium_core.h" +#include "sodium_crypto_core_hchacha20.h" +#include "sodium_crypto_onetimeauth_poly1305.h" +#include "sodium_crypto_secretbox_xchacha20poly1305.h" +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_private_common.h" +#include "sodium_utils.h" + +#define crypto_secretbox_xchacha20poly1305_ZEROBYTES 32U + +int +crypto_secretbox_xchacha20poly1305_detached(unsigned char *c, + unsigned char *mac, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, + const unsigned char *k) +{ + crypto_onetimeauth_poly1305_state state; + unsigned char block0[64U]; + unsigned char subkey[crypto_stream_chacha20_KEYBYTES]; + unsigned long long i; + unsigned long long mlen0; + + crypto_core_hchacha20(subkey, n, k, NULL); + + if (((uintptr_t) c > (uintptr_t) m && + (uintptr_t) c - (uintptr_t) m < mlen) || + ((uintptr_t) m > (uintptr_t) c && + (uintptr_t) m - (uintptr_t) c < mlen)) { /* LCOV_EXCL_LINE */ + memmove(c, m, mlen); + m = c; + } + memset(block0, 0U, crypto_secretbox_xchacha20poly1305_ZEROBYTES); + COMPILER_ASSERT(64U >= crypto_secretbox_xchacha20poly1305_ZEROBYTES); + mlen0 = mlen; + if (mlen0 > 64U - crypto_secretbox_xchacha20poly1305_ZEROBYTES) { + mlen0 = 64U - crypto_secretbox_xchacha20poly1305_ZEROBYTES; + } + for (i = 0U; i < mlen0; i++) { + block0[i + crypto_secretbox_xchacha20poly1305_ZEROBYTES] = m[i]; + } + crypto_stream_chacha20_xor(block0, block0, + mlen0 + crypto_secretbox_xchacha20poly1305_ZEROBYTES, + n + 16, subkey); + COMPILER_ASSERT(crypto_secretbox_xchacha20poly1305_ZEROBYTES >= + crypto_onetimeauth_poly1305_KEYBYTES); + crypto_onetimeauth_poly1305_init(&state, block0); + + for (i = 0U; i < mlen0; i++) { + c[i] = block0[crypto_secretbox_xchacha20poly1305_ZEROBYTES + i]; + } + sodium_memzero(block0, sizeof block0); + if (mlen > mlen0) { + crypto_stream_chacha20_xor_ic(c + mlen0, m + mlen0, mlen - mlen0, + n + 16, 1U, subkey); + } + sodium_memzero(subkey, sizeof subkey); + + crypto_onetimeauth_poly1305_update(&state, c, mlen); + crypto_onetimeauth_poly1305_final(&state, mac); + sodium_memzero(&state, sizeof state); + + return 0; +} + +int +crypto_secretbox_xchacha20poly1305_easy(unsigned char *c, + const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, + const unsigned char *k) +{ + if (mlen > crypto_secretbox_xchacha20poly1305_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return crypto_secretbox_xchacha20poly1305_detached + (c + crypto_secretbox_xchacha20poly1305_MACBYTES, c, m, mlen, n, k); +} + +int +crypto_secretbox_xchacha20poly1305_open_detached(unsigned char *m, + const unsigned char *c, + const unsigned char *mac, + unsigned long long clen, + const unsigned char *n, + const unsigned char *k) +{ + unsigned char block0[64U]; + unsigned char subkey[crypto_stream_chacha20_KEYBYTES]; + unsigned long long i; + unsigned long long mlen0; + + crypto_core_hchacha20(subkey, n, k, NULL); + crypto_stream_chacha20(block0, crypto_stream_chacha20_KEYBYTES, + n + 16, subkey); + if (crypto_onetimeauth_poly1305_verify(mac, c, clen, block0) != 0) { + sodium_memzero(subkey, sizeof subkey); + return -1; + } + if (m == NULL) { + return 0; + } + if (((uintptr_t) c >= (uintptr_t) m && + (uintptr_t) c - (uintptr_t) m < clen) || + ((uintptr_t) m >= (uintptr_t) c && + (uintptr_t) m - (uintptr_t) c < clen)) { /* LCOV_EXCL_LINE */ + memmove(m, c, clen); + c = m; + } + mlen0 = clen; + if (mlen0 > 64U - crypto_secretbox_xchacha20poly1305_ZEROBYTES) { + mlen0 = 64U - crypto_secretbox_xchacha20poly1305_ZEROBYTES; + } + for (i = 0U; i < mlen0; i++) { + block0[crypto_secretbox_xchacha20poly1305_ZEROBYTES + i] = c[i]; + } + crypto_stream_chacha20_xor(block0, block0, + crypto_secretbox_xchacha20poly1305_ZEROBYTES + mlen0, + n + 16, subkey); + for (i = 0U; i < mlen0; i++) { + m[i] = block0[i + crypto_secretbox_xchacha20poly1305_ZEROBYTES]; + } + if (clen > mlen0) { + crypto_stream_chacha20_xor_ic(m + mlen0, c + mlen0, clen - mlen0, + n + 16, 1U, subkey); + } + sodium_memzero(subkey, sizeof subkey); + + return 0; +} + +int +crypto_secretbox_xchacha20poly1305_open_easy(unsigned char *m, + const unsigned char *c, + unsigned long long clen, + const unsigned char *n, + const unsigned char *k) +{ + if (clen < crypto_secretbox_xchacha20poly1305_MACBYTES) { + return -1; + } + return crypto_secretbox_xchacha20poly1305_open_detached + (m, c + crypto_secretbox_xchacha20poly1305_MACBYTES, c, + clen - crypto_secretbox_xchacha20poly1305_MACBYTES, n, k); +} + +size_t +crypto_secretbox_xchacha20poly1305_keybytes(void) +{ + return crypto_secretbox_xchacha20poly1305_KEYBYTES; +} + +size_t +crypto_secretbox_xchacha20poly1305_noncebytes(void) +{ + return crypto_secretbox_xchacha20poly1305_NONCEBYTES; +} + +size_t +crypto_secretbox_xchacha20poly1305_macbytes(void) +{ + return crypto_secretbox_xchacha20poly1305_MACBYTES; +} + +size_t +crypto_secretbox_xchacha20poly1305_messagebytes_max(void) +{ + return crypto_secretbox_xchacha20poly1305_MESSAGEBYTES_MAX; +} diff --git a/sodium/sodium_secretbox_xsalsa20poly1305.c b/sodium/sodium_secretbox_xsalsa20poly1305.c new file mode 100644 index 00000000..825dc5b1 --- /dev/null +++ b/sodium/sodium_secretbox_xsalsa20poly1305.c @@ -0,0 +1,89 @@ +#include "sodium_crypto_onetimeauth_poly1305.h" +#include "sodium_crypto_secretbox_xsalsa20poly1305.h" +#include "sodium_crypto_stream_xsalsa20.h" +#include "sodium_randombytes.h" + +int +crypto_secretbox_xsalsa20poly1305(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, + const unsigned char *k) +{ + int i; + + if (mlen < 32) { + return -1; + } + crypto_stream_xsalsa20_xor(c, m, mlen, n, k); + crypto_onetimeauth_poly1305(c + 16, c + 32, mlen - 32, c); + for (i = 0; i < 16; ++i) { + c[i] = 0; + } + return 0; +} + +int +crypto_secretbox_xsalsa20poly1305_open(unsigned char *m, const unsigned char *c, + unsigned long long clen, + const unsigned char *n, + const unsigned char *k) +{ + unsigned char subkey[32]; + int i; + + if (clen < 32) { + return -1; + } + crypto_stream_xsalsa20(subkey, 32, n, k); + if (crypto_onetimeauth_poly1305_verify(c + 16, c + 32, + clen - 32, subkey) != 0) { + return -1; + } + crypto_stream_xsalsa20_xor(m, c, clen, n, k); + for (i = 0; i < 32; ++i) { + m[i] = 0; + } + return 0; +} + +size_t +crypto_secretbox_xsalsa20poly1305_keybytes(void) +{ + return crypto_secretbox_xsalsa20poly1305_KEYBYTES; +} + +size_t +crypto_secretbox_xsalsa20poly1305_noncebytes(void) +{ + return crypto_secretbox_xsalsa20poly1305_NONCEBYTES; +} + +size_t +crypto_secretbox_xsalsa20poly1305_zerobytes(void) +{ + return crypto_secretbox_xsalsa20poly1305_ZEROBYTES; +} + +size_t +crypto_secretbox_xsalsa20poly1305_boxzerobytes(void) +{ + return crypto_secretbox_xsalsa20poly1305_BOXZEROBYTES; +} + +size_t +crypto_secretbox_xsalsa20poly1305_macbytes(void) +{ + return crypto_secretbox_xsalsa20poly1305_MACBYTES; +} + +size_t +crypto_secretbox_xsalsa20poly1305_messagebytes_max(void) +{ + return crypto_secretbox_xsalsa20poly1305_MESSAGEBYTES_MAX; +} + +void +crypto_secretbox_xsalsa20poly1305_keygen(unsigned char k[crypto_secretbox_xsalsa20poly1305_KEYBYTES]) +{ + randombytes_buf(k, crypto_secretbox_xsalsa20poly1305_KEYBYTES); +} diff --git a/sodium/sodium_sign_ed25519.c b/sodium/sodium_sign_ed25519.c new file mode 100644 index 00000000..54f93374 --- /dev/null +++ b/sodium/sodium_sign_ed25519.c @@ -0,0 +1,97 @@ + +#include + +#include "sodium_crypto_hash_sha512.h" +#include "sodium_crypto_sign_ed25519.h" +#include "sodium_ref10_sign_ed25519.h" + +size_t +crypto_sign_ed25519ph_statebytes(void) +{ + return sizeof(crypto_sign_ed25519ph_state); +} + +size_t +crypto_sign_ed25519_bytes(void) +{ + return crypto_sign_ed25519_BYTES; +} + +size_t +crypto_sign_ed25519_seedbytes(void) +{ + return crypto_sign_ed25519_SEEDBYTES; +} + +size_t +crypto_sign_ed25519_publickeybytes(void) +{ + return crypto_sign_ed25519_PUBLICKEYBYTES; +} + +size_t +crypto_sign_ed25519_secretkeybytes(void) +{ + return crypto_sign_ed25519_SECRETKEYBYTES; +} + +size_t +crypto_sign_ed25519_messagebytes_max(void) +{ + return crypto_sign_ed25519_MESSAGEBYTES_MAX; +} + +int +crypto_sign_ed25519_sk_to_seed(unsigned char *seed, const unsigned char *sk) +{ + memmove(seed, sk, crypto_sign_ed25519_SEEDBYTES); + + return 0; +} + +int +crypto_sign_ed25519_sk_to_pk(unsigned char *pk, const unsigned char *sk) +{ + memmove(pk, sk + crypto_sign_ed25519_SEEDBYTES, + crypto_sign_ed25519_PUBLICKEYBYTES); + return 0; +} + +int +crypto_sign_ed25519ph_init(crypto_sign_ed25519ph_state *state) +{ + crypto_hash_sha512_init(&state->hs); + return 0; +} + +int +crypto_sign_ed25519ph_update(crypto_sign_ed25519ph_state *state, + const unsigned char *m, unsigned long long mlen) +{ + return crypto_hash_sha512_update(&state->hs, m, mlen); +} + +int +crypto_sign_ed25519ph_final_create(crypto_sign_ed25519ph_state *state, + unsigned char *sig, + unsigned long long *siglen_p, + const unsigned char *sk) +{ + unsigned char ph[crypto_hash_sha512_BYTES]; + + crypto_hash_sha512_final(&state->hs, ph); + + return _crypto_sign_ed25519_detached(sig, siglen_p, ph, sizeof ph, sk, 1); +} + +int +crypto_sign_ed25519ph_final_verify(crypto_sign_ed25519ph_state *state, + const unsigned char *sig, + const unsigned char *pk) +{ + unsigned char ph[crypto_hash_sha512_BYTES]; + + crypto_hash_sha512_final(&state->hs, ph); + + return _crypto_sign_ed25519_verify_detached(sig, ph, sizeof ph, pk, 1); +} diff --git a/sodium/sodium_stream_chacha20.c b/sodium/sodium_stream_chacha20.c new file mode 100644 index 00000000..1214071e --- /dev/null +++ b/sodium/sodium_stream_chacha20.c @@ -0,0 +1,194 @@ +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_core.h" +#include "sodium_private_chacha20_ietf_ext.h" +#include "sodium_private_common.h" +#include "sodium_private_implementations.h" +#include "sodium_randombytes.h" +#include "sodium_runtime.h" +#include "sodium_stream_chacha20.h" +#include + +#include "sodium_chacha20-ref.h" +#if defined(HAVE_AVX2INTRIN_H) && defined(HAVE_EMMINTRIN_H) && \ + defined(HAVE_TMMINTRIN_H) && defined(HAVE_SMMINTRIN_H) +# include "sodium_dolbeau_chacha20-avx2.h" +#endif +#if defined(HAVE_EMMINTRIN_H) && defined(HAVE_TMMINTRIN_H) +# include "sodium_dolbeau_chacha20-ssse3.h" +#endif + +static const crypto_stream_chacha20_implementation *implementation = + &crypto_stream_chacha20_ref_implementation; + +size_t +crypto_stream_chacha20_keybytes(void) { + return crypto_stream_chacha20_KEYBYTES; +} + +size_t +crypto_stream_chacha20_noncebytes(void) { + return crypto_stream_chacha20_NONCEBYTES; +} + +size_t +crypto_stream_chacha20_messagebytes_max(void) +{ + return crypto_stream_chacha20_MESSAGEBYTES_MAX; +} + +size_t +crypto_stream_chacha20_ietf_keybytes(void) { + return crypto_stream_chacha20_ietf_KEYBYTES; +} + +size_t +crypto_stream_chacha20_ietf_noncebytes(void) { + return crypto_stream_chacha20_ietf_NONCEBYTES; +} + +size_t +crypto_stream_chacha20_ietf_messagebytes_max(void) +{ + return crypto_stream_chacha20_ietf_MESSAGEBYTES_MAX; +} + +int +crypto_stream_chacha20(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k) +{ + if (clen > crypto_stream_chacha20_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return implementation->stream(c, clen, n, k); +} + +int +crypto_stream_chacha20_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, uint64_t ic, + const unsigned char *k) +{ + if (mlen > crypto_stream_chacha20_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return implementation->stream_xor_ic(c, m, mlen, n, ic, k); +} + +int +crypto_stream_chacha20_xor(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *k) +{ + if (mlen > crypto_stream_chacha20_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return implementation->stream_xor_ic(c, m, mlen, n, 0U, k); +} + +int +crypto_stream_chacha20_ietf_ext(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k) +{ + if (clen > crypto_stream_chacha20_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return implementation->stream_ietf_ext(c, clen, n, k); +} + +int +crypto_stream_chacha20_ietf_ext_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, uint32_t ic, + const unsigned char *k) +{ + if (mlen > crypto_stream_chacha20_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return implementation->stream_ietf_ext_xor_ic(c, m, mlen, n, ic, k); +} + +static int +crypto_stream_chacha20_ietf_ext_xor(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *k) +{ + if (mlen > crypto_stream_chacha20_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return implementation->stream_ietf_ext_xor_ic(c, m, mlen, n, 0U, k); +} + +int +crypto_stream_chacha20_ietf(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k) +{ + if (clen > crypto_stream_chacha20_ietf_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return crypto_stream_chacha20_ietf_ext(c, clen, n, k); +} + +int +crypto_stream_chacha20_ietf_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, uint32_t ic, + const unsigned char *k) +{ + if ((unsigned long long) ic > + (64ULL * (1ULL << 32)) / 64ULL - (mlen + 63ULL) / 64ULL) { + sodium_misuse(); + } + return crypto_stream_chacha20_ietf_ext_xor_ic(c, m, mlen, n, ic, k); +} + +int +crypto_stream_chacha20_ietf_xor(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *k) +{ + if (mlen > crypto_stream_chacha20_ietf_MESSAGEBYTES_MAX) { + sodium_misuse(); + } + return crypto_stream_chacha20_ietf_ext_xor(c, m, mlen, n, k); +} + +void +crypto_stream_chacha20_ietf_keygen(unsigned char k[crypto_stream_chacha20_ietf_KEYBYTES]) +{ + randombytes_buf(k, crypto_stream_chacha20_ietf_KEYBYTES); +} + +void +crypto_stream_chacha20_keygen(unsigned char k[crypto_stream_chacha20_KEYBYTES]) +{ + randombytes_buf(k, crypto_stream_chacha20_KEYBYTES); +} + +int +_crypto_stream_chacha20_pick_best_implementation(void) +{ + implementation = &crypto_stream_chacha20_ref_implementation; +#if defined(HAVE_AVX2INTRIN_H) && defined(HAVE_EMMINTRIN_H) && \ + defined(HAVE_TMMINTRIN_H) && defined(HAVE_SMMINTRIN_H) + if (sodium_runtime_has_avx2()) { + #if NETCODE_CRYPTO_LOGS + printf( "chacha20 -> avx2\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + implementation = &crypto_stream_chacha20_dolbeau_avx2_implementation; + return 0; + } +#endif +#if defined(HAVE_EMMINTRIN_H) && defined(HAVE_TMMINTRIN_H) + if (sodium_runtime_has_ssse3()) { + #if NETCODE_CRYPTO_LOGS + printf( "chacha20 -> ssse3\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + implementation = &crypto_stream_chacha20_dolbeau_ssse3_implementation; + return 0; + } +#endif + #if NETCODE_CRYPTO_LOGS + printf( "chacha20 -> ref\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + return 0; +} diff --git a/sodium/sodium_stream_chacha20.h b/sodium/sodium_stream_chacha20.h new file mode 100644 index 00000000..40f782f4 --- /dev/null +++ b/sodium/sodium_stream_chacha20.h @@ -0,0 +1,22 @@ + +#ifndef stream_chacha20_H +#define stream_chacha20_H + +#include + +typedef struct crypto_stream_chacha20_implementation { + int (*stream)(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k); + int (*stream_ietf_ext)(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k); + int (*stream_xor_ic)(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, uint64_t ic, + const unsigned char *k); + int (*stream_ietf_ext_xor_ic)(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, uint32_t ic, + const unsigned char *k); +} crypto_stream_chacha20_implementation; + +#endif diff --git a/sodium/sodium_stream_salsa20.c b/sodium/sodium_stream_salsa20.c new file mode 100644 index 00000000..03c8a455 --- /dev/null +++ b/sodium/sodium_stream_salsa20.c @@ -0,0 +1,114 @@ +#include "sodium_crypto_stream_salsa20.h" +#include "sodium_private_common.h" +#include "sodium_private_implementations.h" +#include "sodium_randombytes.h" +#include "sodium_runtime.h" +#include "sodium_stream_salsa20.h" +#include + +#ifdef HAVE_AMD64_ASM +# include "sodium_salsa20-xmm6.h" +#else +# include "sodium_salsa20-ref.h" +#endif +#if !defined(HAVE_AMD64_ASM) && defined(HAVE_EMMINTRIN_H) +# include "sodium_xmm6int_salsa20-sse2.h" +#endif +#if defined(HAVE_AVX2INTRIN_H) && defined(HAVE_EMMINTRIN_H) && \ + defined(HAVE_TMMINTRIN_H) && defined(HAVE_SMMINTRIN_H) +# include "sodium_xmm6int_salsa20-avx2.h" +#endif + +#if HAVE_AMD64_ASM +static const crypto_stream_salsa20_implementation *implementation = + &crypto_stream_salsa20_xmm6_implementation; +#else +static const crypto_stream_salsa20_implementation *implementation = + &crypto_stream_salsa20_ref_implementation; +#endif + +size_t +crypto_stream_salsa20_keybytes(void) +{ + return crypto_stream_salsa20_KEYBYTES; +} + +size_t +crypto_stream_salsa20_noncebytes(void) +{ + return crypto_stream_salsa20_NONCEBYTES; +} + +size_t +crypto_stream_salsa20_messagebytes_max(void) +{ + return crypto_stream_salsa20_MESSAGEBYTES_MAX; +} + +int +crypto_stream_salsa20(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k) +{ + return implementation->stream(c, clen, n, k); +} + +int +crypto_stream_salsa20_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, uint64_t ic, + const unsigned char *k) +{ + return implementation->stream_xor_ic(c, m, mlen, n, ic, k); +} + +int +crypto_stream_salsa20_xor(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *k) +{ + return implementation->stream_xor_ic(c, m, mlen, n, 0U, k); +} + +void +crypto_stream_salsa20_keygen(unsigned char k[crypto_stream_salsa20_KEYBYTES]) +{ + randombytes_buf(k, crypto_stream_salsa20_KEYBYTES); +} + +int +_crypto_stream_salsa20_pick_best_implementation(void) +{ +#if defined(HAVE_AVX2INTRIN_H) && defined(HAVE_EMMINTRIN_H) && \ + defined(HAVE_TMMINTRIN_H) && defined(HAVE_SMMINTRIN_H) + if (sodium_runtime_has_avx2()) { + #if NETCODE_CRYPTO_LOGS + printf( "salsa20 -> avx2\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + implementation = &crypto_stream_salsa20_xmm6int_avx2_implementation; + return 0; + } +#endif +#if !defined(HAVE_AMD64_ASM) && defined(HAVE_EMMINTRIN_H) + if (sodium_runtime_has_sse2()) { + #if NETCODE_CRYPTO_LOGS + printf( "salsa20 -> sse2\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + implementation = &crypto_stream_salsa20_xmm6int_sse2_implementation; + return 0; + } +#endif + +#ifdef HAVE_AMD64_ASM + #if NETCODE_CRYPTO_LOGS + printf( "salsa20 -> xmm6\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + implementation = &crypto_stream_salsa20_xmm6_implementation; +#else + #if NETCODE_CRYPTO_LOGS + printf( "salsa20 -> ref\n" ); + #endif // #if NETCODE_CRYPTO_LOGS + implementation = &crypto_stream_salsa20_ref_implementation; +#endif + + return 0; /* LCOV_EXCL_LINE */ +} diff --git a/sodium/sodium_stream_salsa20.h b/sodium/sodium_stream_salsa20.h new file mode 100644 index 00000000..1949d381 --- /dev/null +++ b/sodium/sodium_stream_salsa20.h @@ -0,0 +1,16 @@ + +#ifndef stream_salsa20_H +#define stream_salsa20_H + +#include + +typedef struct crypto_stream_salsa20_implementation { + int (*stream)(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k); + int (*stream_xor_ic)(unsigned char *c, const unsigned char *m, + unsigned long long mlen, + const unsigned char *n, uint64_t ic, + const unsigned char *k); +} crypto_stream_salsa20_implementation; + +#endif diff --git a/sodium/sodium_stream_xchacha20.c b/sodium/sodium_stream_xchacha20.c new file mode 100644 index 00000000..a1de213a --- /dev/null +++ b/sodium/sodium_stream_xchacha20.c @@ -0,0 +1,69 @@ + +#include + +#include "sodium_crypto_core_hchacha20.h" +#include "sodium_crypto_stream_chacha20.h" +#include "sodium_crypto_stream_xchacha20.h" +#include "sodium_private_common.h" +#include "sodium_randombytes.h" + +size_t +crypto_stream_xchacha20_keybytes(void) +{ + return crypto_stream_xchacha20_KEYBYTES; +} + +size_t +crypto_stream_xchacha20_noncebytes(void) +{ + return crypto_stream_xchacha20_NONCEBYTES; +} + +size_t +crypto_stream_xchacha20_messagebytes_max(void) +{ + return crypto_stream_xchacha20_MESSAGEBYTES_MAX; +} + +int +crypto_stream_xchacha20(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k) +{ + unsigned char k2[crypto_core_hchacha20_OUTPUTBYTES]; + + crypto_core_hchacha20(k2, n, k, NULL); + COMPILER_ASSERT(crypto_stream_chacha20_KEYBYTES <= sizeof k2); + COMPILER_ASSERT(crypto_stream_chacha20_NONCEBYTES == + crypto_stream_xchacha20_NONCEBYTES - + crypto_core_hchacha20_INPUTBYTES); + + return crypto_stream_chacha20(c, clen, n + crypto_core_hchacha20_INPUTBYTES, + k2); +} + +int +crypto_stream_xchacha20_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + uint64_t ic, const unsigned char *k) +{ + unsigned char k2[crypto_core_hchacha20_OUTPUTBYTES]; + + crypto_core_hchacha20(k2, n, k, NULL); + return crypto_stream_chacha20_xor_ic( + c, m, mlen, n + crypto_core_hchacha20_INPUTBYTES, ic, k2); +} + +int +crypto_stream_xchacha20_xor(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *k) +{ + return crypto_stream_xchacha20_xor_ic(c, m, mlen, n, 0U, k); +} + +void +crypto_stream_xchacha20_keygen( + unsigned char k[crypto_stream_xchacha20_KEYBYTES]) +{ + randombytes_buf(k, crypto_stream_xchacha20_KEYBYTES); +} diff --git a/sodium/sodium_stream_xsalsa20.c b/sodium/sodium_stream_xsalsa20.c new file mode 100644 index 00000000..4cc109ff --- /dev/null +++ b/sodium/sodium_stream_xsalsa20.c @@ -0,0 +1,66 @@ +#include "sodium_crypto_core_hsalsa20.h" +#include "sodium_crypto_stream_salsa20.h" +#include "sodium_crypto_stream_xsalsa20.h" +#include "sodium_randombytes.h" +#include "sodium_utils.h" + +int +crypto_stream_xsalsa20(unsigned char *c, unsigned long long clen, + const unsigned char *n, const unsigned char *k) +{ + unsigned char subkey[32]; + int ret; + + crypto_core_hsalsa20(subkey, n, k, NULL); + ret = crypto_stream_salsa20(c, clen, n + 16, subkey); + sodium_memzero(subkey, sizeof subkey); + + return ret; +} + +int +crypto_stream_xsalsa20_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + uint64_t ic, const unsigned char *k) +{ + unsigned char subkey[32]; + int ret; + + crypto_core_hsalsa20(subkey, n, k, NULL); + ret = crypto_stream_salsa20_xor_ic(c, m, mlen, n + 16, ic, subkey); + sodium_memzero(subkey, sizeof subkey); + + return ret; +} + +int +crypto_stream_xsalsa20_xor(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, + const unsigned char *k) +{ + return crypto_stream_xsalsa20_xor_ic(c, m, mlen, n, 0ULL, k); +} + +size_t +crypto_stream_xsalsa20_keybytes(void) +{ + return crypto_stream_xsalsa20_KEYBYTES; +} + +size_t +crypto_stream_xsalsa20_noncebytes(void) +{ + return crypto_stream_xsalsa20_NONCEBYTES; +} + +size_t +crypto_stream_xsalsa20_messagebytes_max(void) +{ + return crypto_stream_xsalsa20_MESSAGEBYTES_MAX; +} + +void +crypto_stream_xsalsa20_keygen(unsigned char k[crypto_stream_xsalsa20_KEYBYTES]) +{ + randombytes_buf(k, crypto_stream_xsalsa20_KEYBYTES); +} diff --git a/sodium/sodium_utils.c b/sodium/sodium_utils.c new file mode 100644 index 00000000..2d1fe5a9 --- /dev/null +++ b/sodium/sodium_utils.c @@ -0,0 +1,795 @@ +#ifndef __STDC_WANT_LIB_EXT1__ +# define __STDC_WANT_LIB_EXT1__ 1 +#endif +#include +#include +#include +#if !defined(__ORBIS__) && !defined(__PROSPERO__) +#include +#endif +#include +#include +#include +#include + +#ifdef HAVE_SYS_MMAN_H +# include +#endif + +#ifdef _WIN32 +# include +# if !defined(_XBOX_ONE) && !defined(_GAMING_XBOX) +# include +# endif +#else +# include +#endif + +#ifndef HAVE_C_VARARRAYS +# ifdef HAVE_ALLOCA_H +# include +# elif !defined(alloca) +# if defined(__clang__) || defined(__GNUC__) +# define alloca __builtin_alloca +# elif defined _AIX +# define alloca __alloca +# elif defined _MSC_VER +# include +# define alloca _alloca +# else +# include +# ifdef __cplusplus +extern "C" +# endif +void *alloca (size_t); +# endif +# endif +#endif + +#include "sodium_core.h" +#include "sodium_randombytes.h" +#include "sodium_utils.h" +#include "sodium_private_common.h" + +#ifndef ENOSYS +# define ENOSYS ENXIO +#endif + +#if defined(_WIN32) && \ + (!defined(WINAPI_FAMILY) || WINAPI_FAMILY == WINAPI_FAMILY_DESKTOP_APP) +# define WINAPI_DESKTOP +#endif + +#define CANARY_SIZE 16U +#define GARBAGE_VALUE 0xdb + +#ifndef MAP_NOCORE +# define MAP_NOCORE 0 +#endif +#if !defined(MAP_ANON) && defined(MAP_ANONYMOUS) +# define MAP_ANON MAP_ANONYMOUS +#endif +#if defined(WINAPI_DESKTOP) || (defined(MAP_ANON) && defined(HAVE_MMAP)) || \ + defined(HAVE_POSIX_MEMALIGN) +# define HAVE_ALIGNED_MALLOC +#endif +#if defined(HAVE_MPROTECT) && \ + !(defined(PROT_NONE) && defined(PROT_READ) && defined(PROT_WRITE)) +# undef HAVE_MPROTECT +#endif +#if defined(HAVE_ALIGNED_MALLOC) && \ + (defined(WINAPI_DESKTOP) || defined(HAVE_MPROTECT)) +# define HAVE_PAGE_PROTECTION +#endif +#if !defined(MADV_DODUMP) && defined(MADV_CORE) +# define MADV_DODUMP MADV_CORE +# define MADV_DONTDUMP MADV_NOCORE +#endif + +#ifdef HAVE_ALIGNED_MALLOC +static size_t page_size; +#endif // #if HAVE_ALIGNED_MALLOC +static unsigned char canary[CANARY_SIZE]; + +/* LCOV_EXCL_START */ +#ifdef HAVE_WEAK_SYMBOLS +__attribute__((weak)) void +_sodium_dummy_symbol_to_prevent_memzero_lto(void *const pnt, + const size_t len); +__attribute__((weak)) void +_sodium_dummy_symbol_to_prevent_memzero_lto(void *const pnt, + const size_t len) +{ + (void) pnt; /* LCOV_EXCL_LINE */ + (void) len; /* LCOV_EXCL_LINE */ +} +#endif +/* LCOV_EXCL_STOP */ + +void +sodium_memzero(void *const pnt, const size_t len) +{ +#ifdef _WIN32 + SecureZeroMemory(pnt, len); +#elif defined(HAVE_MEMSET_S) + if (len > 0U && memset_s(pnt, (rsize_t) len, 0, (rsize_t) len) != 0) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } +#elif defined(HAVE_EXPLICIT_BZERO) + explicit_bzero(pnt, len); +#elif defined(HAVE_EXPLICIT_MEMSET) + explicit_memset(pnt, 0, len); +#elif HAVE_WEAK_SYMBOLS + memset(pnt, 0, len); + _sodium_dummy_symbol_to_prevent_memzero_lto(pnt, len); +# ifdef HAVE_INLINE_ASM + __asm__ __volatile__ ("" : : "r"(pnt) : "memory"); +# endif +#else + volatile unsigned char *volatile pnt_ = + (volatile unsigned char *volatile) pnt; + size_t i = (size_t) 0U; + + while (i < len) { + pnt_[i++] = 0U; + } +#endif +} + +void +sodium_stackzero(const size_t len) +{ + (void) len; +#ifdef HAVE_C_VARARRAYS + unsigned char fodder[len]; + sodium_memzero(fodder, len); +#elif HAVE_ALLOCA + sodium_memzero(alloca(len), len); +#endif +} + +#ifdef HAVE_WEAK_SYMBOLS +__attribute__((weak)) void +_sodium_dummy_symbol_to_prevent_memcmp_lto(const unsigned char *b1, + const unsigned char *b2, + const size_t len); +__attribute__((weak)) void +_sodium_dummy_symbol_to_prevent_memcmp_lto(const unsigned char *b1, + const unsigned char *b2, + const size_t len) +{ + (void) b1; + (void) b2; + (void) len; +} +#endif + +int +sodium_memcmp(const void *const b1_, const void *const b2_, size_t len) +{ +#ifdef HAVE_WEAK_SYMBOLS + const unsigned char *b1 = (const unsigned char *) b1_; + const unsigned char *b2 = (const unsigned char *) b2_; +#else + const volatile unsigned char * b1 = + (const volatile unsigned char *) b1_; + const volatile unsigned char *volatile b2 = + (const volatile unsigned char *) b2_; +#endif + size_t i; + volatile unsigned char d = 0U; + +#if HAVE_WEAK_SYMBOLS + _sodium_dummy_symbol_to_prevent_memcmp_lto(b1, b2, len); +#endif + for (i = 0U; i < len; i++) { + d |= b1[i] ^ b2[i]; + } + return (1 & ((d - 1) >> 8)) - 1; +} + +#ifdef HAVE_WEAK_SYMBOLS +__attribute__((weak)) void +_sodium_dummy_symbol_to_prevent_compare_lto(const unsigned char *b1, + const unsigned char *b2, + const size_t len); +__attribute__((weak)) void +_sodium_dummy_symbol_to_prevent_compare_lto(const unsigned char *b1, + const unsigned char *b2, + const size_t len) +{ + (void) b1; + (void) b2; + (void) len; +} +#endif + +int +sodium_compare(const unsigned char *b1_, const unsigned char *b2_, size_t len) +{ +#ifdef HAVE_WEAK_SYMBOLS + const unsigned char *b1 = b1_; + const unsigned char *b2 = b2_; +#else + const volatile unsigned char * b1 = + (const volatile unsigned char *) b1_; + const volatile unsigned char * b2 = + (const volatile unsigned char *) b2_; +#endif + size_t i; + volatile unsigned char gt = 0U; + volatile unsigned char eq = 1U; + uint16_t x1, x2; + +#if HAVE_WEAK_SYMBOLS + _sodium_dummy_symbol_to_prevent_compare_lto(b1, b2, len); +#endif + i = len; + while (i != 0U) { + i--; + x1 = b1[i]; + x2 = b2[i]; + gt |= ((x2 - x1) >> 8) & eq; + eq &= ((x2 ^ x1) - 1) >> 8; + } + return (int) (gt + gt + eq) - 1; +} + +int +sodium_is_zero(const unsigned char *n, const size_t nlen) +{ + size_t i; + volatile unsigned char d = 0U; + + for (i = 0U; i < nlen; i++) { + d |= n[i]; + } + return 1 & ((d - 1) >> 8); +} + +void +sodium_increment(unsigned char *n, const size_t nlen) +{ + size_t i = 0U; + uint_fast16_t c = 1U; + +#ifdef HAVE_AMD64_ASM + uint64_t t64, t64_2; + uint32_t t32; + + if (nlen == 12U) { + __asm__ __volatile__( + "xorq %[t64], %[t64] \n" + "xorl %[t32], %[t32] \n" + "stc \n" + "adcq %[t64], (%[out]) \n" + "adcl %[t32], 8(%[out]) \n" + : [t64] "=&r"(t64), [t32] "=&r"(t32) + : [out] "D"(n) + : "memory", "flags", "cc"); + return; + } else if (nlen == 24U) { + __asm__ __volatile__( + "movq $1, %[t64] \n" + "xorq %[t64_2], %[t64_2] \n" + "addq %[t64], (%[out]) \n" + "adcq %[t64_2], 8(%[out]) \n" + "adcq %[t64_2], 16(%[out]) \n" + : [t64] "=&r"(t64), [t64_2] "=&r"(t64_2) + : [out] "D"(n) + : "memory", "flags", "cc"); + return; + } else if (nlen == 8U) { + __asm__ __volatile__("incq (%[out]) \n" + : + : [out] "D"(n) + : "memory", "flags", "cc"); + return; + } +#endif + for (; i < nlen; i++) { + c += (uint_fast16_t) n[i]; + n[i] = (unsigned char) c; + c >>= 8; + } +} + +void +sodium_add(unsigned char *a, const unsigned char *b, const size_t len) +{ + size_t i; + uint_fast16_t c = 0U; + +#ifdef HAVE_AMD64_ASM + uint64_t t64, t64_2, t64_3; + uint32_t t32; + + if (len == 12U) { + __asm__ __volatile__( + "movq (%[in]), %[t64] \n" + "movl 8(%[in]), %[t32] \n" + "addq %[t64], (%[out]) \n" + "adcl %[t32], 8(%[out]) \n" + : [t64] "=&r"(t64), [t32] "=&r"(t32) + : [in] "S"(b), [out] "D"(a) + : "memory", "flags", "cc"); + return; + } else if (len == 24U) { + __asm__ __volatile__( + "movq (%[in]), %[t64] \n" + "movq 8(%[in]), %[t64_2] \n" + "movq 16(%[in]), %[t64_3] \n" + "addq %[t64], (%[out]) \n" + "adcq %[t64_2], 8(%[out]) \n" + "adcq %[t64_3], 16(%[out]) \n" + : [t64] "=&r"(t64), [t64_2] "=&r"(t64_2), [t64_3] "=&r"(t64_3) + : [in] "S"(b), [out] "D"(a) + : "memory", "flags", "cc"); + return; + } else if (len == 8U) { + __asm__ __volatile__( + "movq (%[in]), %[t64] \n" + "addq %[t64], (%[out]) \n" + : [t64] "=&r"(t64) + : [in] "S"(b), [out] "D"(a) + : "memory", "flags", "cc"); + return; + } +#endif + for (i = 0U; i < len; i++) { + c += (uint_fast16_t) a[i] + (uint_fast16_t) b[i]; + a[i] = (unsigned char) c; + c >>= 8; + } +} + +void +sodium_sub(unsigned char *a, const unsigned char *b, const size_t len) +{ + uint_fast16_t c = 0U; + size_t i; + +#ifdef HAVE_AMD64_ASM + uint64_t t64_1, t64_2, t64_3, t64_4; + uint64_t t64_5, t64_6, t64_7, t64_8; + uint32_t t32; + + if (len == 64U) { + __asm__ __volatile__( + "movq (%[in]), %[t64_1] \n" + "movq 8(%[in]), %[t64_2] \n" + "movq 16(%[in]), %[t64_3] \n" + "movq 24(%[in]), %[t64_4] \n" + "movq 32(%[in]), %[t64_5] \n" + "movq 40(%[in]), %[t64_6] \n" + "movq 48(%[in]), %[t64_7] \n" + "movq 56(%[in]), %[t64_8] \n" + "subq %[t64_1], (%[out]) \n" + "sbbq %[t64_2], 8(%[out]) \n" + "sbbq %[t64_3], 16(%[out]) \n" + "sbbq %[t64_4], 24(%[out]) \n" + "sbbq %[t64_5], 32(%[out]) \n" + "sbbq %[t64_6], 40(%[out]) \n" + "sbbq %[t64_7], 48(%[out]) \n" + "sbbq %[t64_8], 56(%[out]) \n" + : [t64_1] "=&r"(t64_1), [t64_2] "=&r"(t64_2), [t64_3] "=&r"(t64_3), [t64_4] "=&r"(t64_4), + [t64_5] "=&r"(t64_5), [t64_6] "=&r"(t64_6), [t64_7] "=&r"(t64_7), [t64_8] "=&r"(t64_8) + : [in] "S"(b), [out] "D"(a) + : "memory", "flags", "cc"); + return; + } +#endif + for (i = 0U; i < len; i++) { + c = (uint_fast16_t) a[i] - (uint_fast16_t) b[i] - c; + a[i] = (unsigned char) c; + c = (c >> 8) & 1U; + } +} + +int +_sodium_alloc_init(void) +{ +#ifdef HAVE_ALIGNED_MALLOC +# if defined(_SC_PAGESIZE) + long page_size_ = sysconf(_SC_PAGESIZE); + if (page_size_ > 0L) { + page_size = (size_t) page_size_; + } +# elif defined(WINAPI_DESKTOP) + SYSTEM_INFO si; + GetSystemInfo(&si); + page_size = (size_t) si.dwPageSize; +# endif + if (page_size < CANARY_SIZE || page_size < sizeof(size_t)) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } +#endif + randombytes_buf(canary, sizeof canary); + + return 0; +} + +int +sodium_mlock(void *const addr, const size_t len) +{ + (void) len; + (void) addr; +#if defined(MADV_DONTDUMP) && defined(HAVE_MADVISE) + (void) madvise(addr, len, MADV_DONTDUMP); +#endif +#ifdef HAVE_MLOCK + return mlock(addr, len); +#elif defined(WINAPI_DESKTOP) + return -(VirtualLock(addr, len) == 0); +#else + errno = ENOSYS; + return -1; +#endif +} + +int +sodium_munlock(void *const addr, const size_t len) +{ + sodium_memzero(addr, len); +#if defined(MADV_DODUMP) && defined(HAVE_MADVISE) + (void) madvise(addr, len, MADV_DODUMP); +#endif +#ifdef HAVE_MLOCK + return munlock(addr, len); +#elif defined(WINAPI_DESKTOP) + return -(VirtualUnlock(addr, len) == 0); +#else + errno = ENOSYS; + return -1; +#endif +} + +static int +_mprotect_noaccess(void *ptr, size_t size) +{ + (void) ptr; + (void) size; +#ifdef HAVE_MPROTECT + return mprotect(ptr, size, PROT_NONE); +#elif defined(WINAPI_DESKTOP) + DWORD old; + return -(VirtualProtect(ptr, size, PAGE_NOACCESS, &old) == 0); +#else + errno = ENOSYS; + return -1; +#endif +} + +static int +_mprotect_readonly(void *ptr, size_t size) +{ + (void) ptr; + (void) size; +#ifdef HAVE_MPROTECT + return mprotect(ptr, size, PROT_READ); +#elif defined(WINAPI_DESKTOP) + DWORD old; + return -(VirtualProtect(ptr, size, PAGE_READONLY, &old) == 0); +#else + errno = ENOSYS; + return -1; +#endif +} + +static int +_mprotect_readwrite(void *ptr, size_t size) +{ + (void) ptr; + (void) size; +#ifdef HAVE_MPROTECT + return mprotect(ptr, size, PROT_READ | PROT_WRITE); +#elif defined(WINAPI_DESKTOP) + DWORD old; + return -(VirtualProtect(ptr, size, PAGE_READWRITE, &old) == 0); +#else + errno = ENOSYS; + return -1; +#endif +} + +#ifdef HAVE_ALIGNED_MALLOC + +__attribute__((noreturn)) static void +_out_of_bounds(void) +{ +# ifdef SIGSEGV + raise(SIGSEGV); +# elif defined(SIGKILL) + raise(SIGKILL); +# endif + abort(); /* not something we want any higher-level API to catch */ +} /* LCOV_EXCL_LINE */ + +static inline size_t +_page_round(const size_t size) +{ + const size_t page_mask = page_size - 1U; + + return (size + page_mask) & ~page_mask; +} + +static __attribute__((malloc)) unsigned char * +_alloc_aligned(const size_t size) +{ + void *ptr; + +# if defined(MAP_ANON) && defined(HAVE_MMAP) + if ((ptr = mmap(NULL, size, PROT_READ | PROT_WRITE, + MAP_ANON | MAP_PRIVATE | MAP_NOCORE, -1, 0)) == + MAP_FAILED) { + ptr = NULL; /* LCOV_EXCL_LINE */ + } /* LCOV_EXCL_LINE */ +# elif defined(HAVE_POSIX_MEMALIGN) + if (posix_memalign(&ptr, page_size, size) != 0) { + ptr = NULL; /* LCOV_EXCL_LINE */ + } /* LCOV_EXCL_LINE */ +# elif defined(WINAPI_DESKTOP) + ptr = VirtualAlloc(NULL, size, MEM_COMMIT | MEM_RESERVE, PAGE_READWRITE); +# else +# error Bug +# endif + return (unsigned char *) ptr; +} + +static void +_free_aligned(unsigned char *const ptr, const size_t size) +{ +# if defined(MAP_ANON) && defined(HAVE_MMAP) + (void) munmap(ptr, size); +# elif defined(HAVE_POSIX_MEMALIGN) + free(ptr); +# elif defined(WINAPI_DESKTOP) + VirtualFree(ptr, 0U, MEM_RELEASE); +# else +# error Bug +#endif +} + +static unsigned char * +_unprotected_ptr_from_user_ptr(void *const ptr) +{ + uintptr_t unprotected_ptr_u; + unsigned char *canary_ptr; + size_t page_mask; + + canary_ptr = ((unsigned char *) ptr) - sizeof canary; + page_mask = page_size - 1U; + unprotected_ptr_u = ((uintptr_t) canary_ptr & (uintptr_t) ~page_mask); + if (unprotected_ptr_u <= page_size * 2U) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } + return (unsigned char *) unprotected_ptr_u; +} + +#endif /* HAVE_ALIGNED_MALLOC */ + +#ifndef HAVE_ALIGNED_MALLOC +static __attribute__((malloc)) void * +_sodium_malloc(const size_t size) +{ + return malloc(size > (size_t) 0U ? size : (size_t) 1U); +} +#else +static __attribute__((malloc)) void * +_sodium_malloc(const size_t size) +{ + void *user_ptr; + unsigned char *base_ptr; + unsigned char *canary_ptr; + unsigned char *unprotected_ptr; + size_t size_with_canary; + size_t total_size; + size_t unprotected_size; + + if (size >= (size_t) SIZE_MAX - page_size * 4U) { + errno = ENOMEM; + return NULL; + } + if (page_size <= sizeof canary || page_size < sizeof unprotected_size) { + sodium_misuse(); /* LCOV_EXCL_LINE */ + } + size_with_canary = (sizeof canary) + size; + unprotected_size = _page_round(size_with_canary); + total_size = page_size + page_size + unprotected_size + page_size; + if ((base_ptr = _alloc_aligned(total_size)) == NULL) { + return NULL; /* LCOV_EXCL_LINE */ + } + unprotected_ptr = base_ptr + page_size * 2U; + _mprotect_noaccess(base_ptr + page_size, page_size); +# ifndef HAVE_PAGE_PROTECTION + memcpy(unprotected_ptr + unprotected_size, canary, sizeof canary); +# endif + _mprotect_noaccess(unprotected_ptr + unprotected_size, page_size); + sodium_mlock(unprotected_ptr, unprotected_size); + canary_ptr = + unprotected_ptr + _page_round(size_with_canary) - size_with_canary; + user_ptr = canary_ptr + sizeof canary; + memcpy(canary_ptr, canary, sizeof canary); + memcpy(base_ptr, &unprotected_size, sizeof unprotected_size); + _mprotect_readonly(base_ptr, page_size); + assert(_unprotected_ptr_from_user_ptr(user_ptr) == unprotected_ptr); + + return user_ptr; +} +#endif /* !HAVE_ALIGNED_MALLOC */ + +__attribute__((malloc)) void * +sodium_malloc(const size_t size) +{ + void *ptr; + + if ((ptr = _sodium_malloc(size)) == NULL) { + return NULL; + } + memset(ptr, (int) GARBAGE_VALUE, size); + + return ptr; +} + +__attribute__((malloc)) void * +sodium_allocarray(size_t count, size_t size) +{ + if (count > (size_t) 0U && size >= (size_t) SIZE_MAX / count) { + errno = ENOMEM; + return NULL; + } + return sodium_malloc(count * size); +} + +#ifndef HAVE_ALIGNED_MALLOC +void +sodium_free(void *ptr) +{ + free(ptr); +} +#else +void +sodium_free(void *ptr) +{ + unsigned char *base_ptr; + unsigned char *canary_ptr; + unsigned char *unprotected_ptr; + size_t total_size; + size_t unprotected_size; + + if (ptr == NULL) { + return; + } + canary_ptr = ((unsigned char *) ptr) - sizeof canary; + unprotected_ptr = _unprotected_ptr_from_user_ptr(ptr); + base_ptr = unprotected_ptr - page_size * 2U; + memcpy(&unprotected_size, base_ptr, sizeof unprotected_size); + total_size = page_size + page_size + unprotected_size + page_size; + _mprotect_readwrite(base_ptr, total_size); + if (sodium_memcmp(canary_ptr, canary, sizeof canary) != 0) { + _out_of_bounds(); + } +# ifndef HAVE_PAGE_PROTECTION + if (sodium_memcmp(unprotected_ptr + unprotected_size, canary, + sizeof canary) != 0) { + _out_of_bounds(); + } +# endif + sodium_munlock(unprotected_ptr, unprotected_size); + _free_aligned(base_ptr, total_size); +} +#endif /* HAVE_ALIGNED_MALLOC */ + +#ifndef HAVE_PAGE_PROTECTION +static int +_sodium_mprotect(void *ptr, int (*cb)(void *ptr, size_t size)) +{ + (void) ptr; + (void) cb; + errno = ENOSYS; + return -1; +} +#else +static int +_sodium_mprotect(void *ptr, int (*cb)(void *ptr, size_t size)) +{ + unsigned char *base_ptr; + unsigned char *unprotected_ptr; + size_t unprotected_size; + + unprotected_ptr = _unprotected_ptr_from_user_ptr(ptr); + base_ptr = unprotected_ptr - page_size * 2U; + memcpy(&unprotected_size, base_ptr, sizeof unprotected_size); + + return cb(unprotected_ptr, unprotected_size); +} +#endif + +int +sodium_mprotect_noaccess(void *ptr) +{ + return _sodium_mprotect(ptr, _mprotect_noaccess); +} + +int +sodium_mprotect_readonly(void *ptr) +{ + return _sodium_mprotect(ptr, _mprotect_readonly); +} + +int +sodium_mprotect_readwrite(void *ptr) +{ + return _sodium_mprotect(ptr, _mprotect_readwrite); +} + +int +sodium_pad(size_t *padded_buflen_p, unsigned char *buf, + size_t unpadded_buflen, size_t blocksize, size_t max_buflen) +{ + unsigned char *tail; + size_t i; + size_t xpadlen; + size_t xpadded_len; + volatile unsigned char mask; + unsigned char barrier_mask; + + if (blocksize <= 0U) { + return -1; + } + xpadlen = blocksize - 1U; + if ((blocksize & (blocksize - 1U)) == 0U) { + xpadlen -= unpadded_buflen & (blocksize - 1U); + } else { + xpadlen -= unpadded_buflen % blocksize; + } + if ((size_t) SIZE_MAX - unpadded_buflen <= xpadlen) { + sodium_misuse(); + } + xpadded_len = unpadded_buflen + xpadlen; + if (xpadded_len >= max_buflen) { + return -1; + } + tail = &buf[xpadded_len]; + if (padded_buflen_p != NULL) { + *padded_buflen_p = xpadded_len + 1U; + } + mask = 0U; + for (i = 0; i < blocksize; i++) { + barrier_mask = (unsigned char) (((i ^ xpadlen) - 1U) + >> ((sizeof(size_t) - 1) * CHAR_BIT)); + *(tail - i) = ((*(tail - i)) & mask) | (0x80 & barrier_mask); + mask |= barrier_mask; + } + return 0; +} + +int +sodium_unpad(size_t *unpadded_buflen_p, const unsigned char *buf, + size_t padded_buflen, size_t blocksize) +{ + const unsigned char *tail; + unsigned char acc = 0U; + unsigned char c; + unsigned char valid = 0U; + volatile size_t pad_len = 0U; + size_t i; + size_t is_barrier; + + if (padded_buflen < blocksize || blocksize <= 0U) { + return -1; + } + tail = &buf[padded_buflen - 1U]; + + for (i = 0U; i < blocksize; i++) { + c = *(tail - i); + is_barrier = + (( (acc - 1U) & (pad_len - 1U) & ((c ^ 0x80) - 1U) ) >> 8) & 1U; + acc |= c; + pad_len |= i & (1U + ~is_barrier); + valid |= (unsigned char) is_barrier; + } + *unpadded_buflen_p = padded_buflen - 1U - pad_len; + + return (int) (valid - 1U); +} diff --git a/windows/sodium/utils.h b/sodium/sodium_utils.h similarity index 56% rename from windows/sodium/utils.h rename to sodium/sodium_utils.h index 0a7aadb4..d9191c0f 100644 --- a/windows/sodium/utils.h +++ b/sodium/sodium_utils.h @@ -4,7 +4,7 @@ #include -#include "export.h" +#include "sodium_export.h" #ifdef __cplusplus extern "C" { @@ -18,8 +18,9 @@ extern "C" { # endif #endif -SODIUM_EXPORT -void sodium_memzero(void * const pnt, const size_t len); +void sodium_memzero(void * const pnt, const size_t len) __attribute__ ((nonnull)); + +void sodium_stackzero(const size_t len); /* * WARNING: sodium_memcmp() must be used to verify if two secret keys @@ -27,9 +28,8 @@ void sodium_memzero(void * const pnt, const size_t len); * It returns 0 if the keys are equal, and -1 if they differ. * This function is not designed for lexicographical comparisons. */ -SODIUM_EXPORT int sodium_memcmp(const void * const b1_, const void * const b2_, size_t len) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); /* * sodium_compare() returns -1 if b1_ < b2_, 1 if b1_ > b2_ and 0 if b1_ == b2_ @@ -37,35 +37,61 @@ int sodium_memcmp(const void * const b1_, const void * const b2_, size_t len) * and counters stored in little-endian format. * However, it is slower than sodium_memcmp(). */ -SODIUM_EXPORT int sodium_compare(const unsigned char *b1_, const unsigned char *b2_, size_t len) - __attribute__ ((warn_unused_result)); + __attribute__ ((warn_unused_result)) __attribute__ ((nonnull)); -SODIUM_EXPORT int sodium_is_zero(const unsigned char *n, const size_t nlen); -SODIUM_EXPORT void sodium_increment(unsigned char *n, const size_t nlen); -SODIUM_EXPORT -void sodium_add(unsigned char *a, const unsigned char *b, const size_t len); +void sodium_add(unsigned char *a, const unsigned char *b, const size_t len) + __attribute__ ((nonnull)); + +void sodium_sub(unsigned char *a, const unsigned char *b, const size_t len) + __attribute__ ((nonnull)); -SODIUM_EXPORT char *sodium_bin2hex(char * const hex, const size_t hex_maxlen, - const unsigned char * const bin, const size_t bin_len); + const unsigned char * const bin, const size_t bin_len) + __attribute__ ((nonnull)); -SODIUM_EXPORT int sodium_hex2bin(unsigned char * const bin, const size_t bin_maxlen, const char * const hex, const size_t hex_len, const char * const ignore, size_t * const bin_len, - const char ** const hex_end); + const char ** const hex_end) + __attribute__ ((nonnull(1, 3))); + +#define sodium_base64_VARIANT_ORIGINAL 1 +#define sodium_base64_VARIANT_ORIGINAL_NO_PADDING 3 +#define sodium_base64_VARIANT_URLSAFE 5 +#define sodium_base64_VARIANT_URLSAFE_NO_PADDING 7 + +/* + * Computes the required length to encode BIN_LEN bytes as a base64 string + * using the given variant. The computed length includes a trailing \0. + */ +#define sodium_base64_ENCODED_LEN(BIN_LEN, VARIANT) \ + (((BIN_LEN) / 3U) * 4U + \ + ((((BIN_LEN) - ((BIN_LEN) / 3U) * 3U) | (((BIN_LEN) - ((BIN_LEN) / 3U) * 3U) >> 1)) & 1U) * \ + (4U - (~((((VARIANT) & 2U) >> 1) - 1U) & (3U - ((BIN_LEN) - ((BIN_LEN) / 3U) * 3U)))) + 1U) -SODIUM_EXPORT -int sodium_mlock(void * const addr, const size_t len); +size_t sodium_base64_encoded_len(const size_t bin_len, const int variant); -SODIUM_EXPORT -int sodium_munlock(void * const addr, const size_t len); +char *sodium_bin2base64(char * const b64, const size_t b64_maxlen, + const unsigned char * const bin, const size_t bin_len, + const int variant) __attribute__ ((nonnull)); + +int sodium_base642bin(unsigned char * const bin, const size_t bin_maxlen, + const char * const b64, const size_t b64_len, + const char * const ignore, size_t * const bin_len, + const char ** const b64_end, const int variant) + __attribute__ ((nonnull(1, 3))); + +int sodium_mlock(void * const addr, const size_t len) + __attribute__ ((nonnull)); + +int sodium_munlock(void * const addr, const size_t len) + __attribute__ ((nonnull)); /* WARNING: sodium_malloc() and sodium_allocarray() are not general-purpose * allocation functions. @@ -100,25 +126,27 @@ int sodium_munlock(void * const addr, const size_t len); * state = sodium_malloc(crypto_generichash_statebytes()); */ -SODIUM_EXPORT void *sodium_malloc(const size_t size) __attribute__ ((malloc)); -SODIUM_EXPORT void *sodium_allocarray(size_t count, size_t size) __attribute__ ((malloc)); -SODIUM_EXPORT void sodium_free(void *ptr); -SODIUM_EXPORT -int sodium_mprotect_noaccess(void *ptr); +int sodium_mprotect_noaccess(void *ptr) __attribute__ ((nonnull)); + +int sodium_mprotect_readonly(void *ptr) __attribute__ ((nonnull)); + +int sodium_mprotect_readwrite(void *ptr) __attribute__ ((nonnull)); -SODIUM_EXPORT -int sodium_mprotect_readonly(void *ptr); +int sodium_pad(size_t *padded_buflen_p, unsigned char *buf, + size_t unpadded_buflen, size_t blocksize, size_t max_buflen) + __attribute__ ((nonnull(2))); -SODIUM_EXPORT -int sodium_mprotect_readwrite(void *ptr); +int sodium_unpad(size_t *unpadded_buflen_p, const unsigned char *buf, + size_t padded_buflen, size_t blocksize) + __attribute__ ((nonnull(2))); /* -------- */ diff --git a/sodium/sodium_verify.c b/sodium/sodium_verify.c new file mode 100644 index 00000000..f7a2d8f0 --- /dev/null +++ b/sodium/sodium_verify.c @@ -0,0 +1,98 @@ + +#include +#include + +#include "sodium_crypto_verify_16.h" +#include "sodium_crypto_verify_32.h" +#include "sodium_crypto_verify_64.h" + +size_t +crypto_verify_16_bytes(void) +{ + return crypto_verify_16_BYTES; +} + +size_t +crypto_verify_32_bytes(void) +{ + return crypto_verify_32_BYTES; +} + +size_t +crypto_verify_64_bytes(void) +{ + return crypto_verify_64_BYTES; +} + +#if defined(HAVE_EMMINTRIN_H) && defined(__SSE2__) + +# ifdef __GNUC__ +# pragma GCC target("sse2") +# endif +# include + +static inline int +crypto_verify_n(const unsigned char *x_, const unsigned char *y_, + const int n) +{ + const __m128i zero = _mm_setzero_si128(); + volatile __m128i v1, v2, z; + volatile int m; + int i; + + const volatile __m128i *volatile x = + (const volatile __m128i *volatile) (const void *) x_; + const volatile __m128i *volatile y = + (const volatile __m128i *volatile) (const void *) y_; + v1 = _mm_loadu_si128((const __m128i *) &x[0]); + v2 = _mm_loadu_si128((const __m128i *) &y[0]); + z = _mm_xor_si128(v1, v2); + for (i = 1; i < n / 16; i++) { + v1 = _mm_loadu_si128((const __m128i *) &x[i]); + v2 = _mm_loadu_si128((const __m128i *) &y[i]); + z = _mm_or_si128(z, _mm_xor_si128(v1, v2)); + } + m = _mm_movemask_epi8(_mm_cmpeq_epi32(z, zero)); + v1 = zero; v2 = zero; z = zero; + + return (int) (((uint32_t) m + 1U) >> 16) - 1; +} + +#else + +static inline int +crypto_verify_n(const unsigned char *x_, const unsigned char *y_, + const int n) +{ + const volatile unsigned char * x = + (const volatile unsigned char *) x_; + const volatile unsigned char * y = + (const volatile unsigned char *) y_; + volatile uint_fast16_t d = 0U; + int i; + + for (i = 0; i < n; i++) { + d |= x[i] ^ y[i]; + } + return (1 & ((d - 1) >> 8)) - 1; +} + +#endif + +int +crypto_verify_16(const unsigned char *x, const unsigned char *y) +{ + return crypto_verify_n(x, y, crypto_verify_16_BYTES); +} + +int +crypto_verify_32(const unsigned char *x, const unsigned char *y) +{ + return crypto_verify_n(x, y, crypto_verify_32_BYTES); +} + +int +crypto_verify_64(const unsigned char *x, const unsigned char *y) +{ + return crypto_verify_n(x, y, crypto_verify_64_BYTES); +} diff --git a/sodium/sodium_version.c b/sodium/sodium_version.c new file mode 100644 index 00000000..a6b0c0ef --- /dev/null +++ b/sodium/sodium_version.c @@ -0,0 +1,30 @@ + +#include "sodium_version.h" + +const char * +sodium_version_string(void) +{ + return SODIUM_VERSION_STRING; +} + +int +sodium_library_version_major(void) +{ + return SODIUM_LIBRARY_VERSION_MAJOR; +} + +int +sodium_library_version_minor(void) +{ + return SODIUM_LIBRARY_VERSION_MINOR; +} + +int +sodium_library_minimal(void) +{ +#ifdef SODIUM_LIBRARY_MINIMAL + return 1; +#else + return 0; +#endif +} diff --git a/windows/sodium/version.h b/sodium/sodium_version.h similarity index 61% rename from windows/sodium/version.h rename to sodium/sodium_version.h index ab8839f7..b7ff3c11 100644 --- a/windows/sodium/version.h +++ b/sodium/sodium_version.h @@ -1,29 +1,23 @@ - #ifndef sodium_version_H #define sodium_version_H -#include "export.h" - -#define SODIUM_VERSION_STRING "1.0.13" +#include "sodium_export.h" -#define SODIUM_LIBRARY_VERSION_MAJOR 9 -#define SODIUM_LIBRARY_VERSION_MINOR 5 +#define SODIUM_VERSION_STRING "1.0.17" +#define SODIUM_LIBRARY_VERSION_MAJOR 10 +#define SODIUM_LIBRARY_VERSION_MINOR 2 #ifdef __cplusplus extern "C" { #endif -SODIUM_EXPORT const char *sodium_version_string(void); -SODIUM_EXPORT int sodium_library_version_major(void); -SODIUM_EXPORT int sodium_library_version_minor(void); -SODIUM_EXPORT int sodium_library_minimal(void); #ifdef __cplusplus diff --git a/sodium/sodium_xmm6int_salsa20-avx2.c b/sodium/sodium_xmm6int_salsa20-avx2.c new file mode 100644 index 00000000..4ab9b93b --- /dev/null +++ b/sodium/sodium_xmm6int_salsa20-avx2.c @@ -0,0 +1,133 @@ + +#include +#include +#include + +#include "sodium_crypto_stream_salsa20.h" +#include "sodium_private_common.h" +#include "sodium_private_sse2_64_32.h" +#include "sodium_utils.h" + +#if defined(HAVE_AVX2INTRIN_H) && defined(HAVE_EMMINTRIN_H) && \ + defined(HAVE_TMMINTRIN_H) && defined(HAVE_SMMINTRIN_H) + +# ifdef __GNUC__ +# pragma GCC target("sse2") +# pragma GCC target("ssse3") +# pragma GCC target("sse4.1") +# pragma GCC target("avx2") +# endif + +#include +#include +#include +#include + +# include "sodium_stream_salsa20.h" +# include "sodium_xmm6int_salsa20-avx2.h" + +# define ROUNDS 20 + +typedef struct salsa_ctx { + uint32_t input[16]; +} salsa_ctx; + +static const int TR[16] = { + 0, 5, 10, 15, 12, 1, 6, 11, 8, 13, 2, 7, 4, 9, 14, 3 +}; + +static void +salsa_keysetup(salsa_ctx *ctx, const uint8_t *k) +{ + ctx->input[TR[1]] = LOAD32_LE(k + 0); + ctx->input[TR[2]] = LOAD32_LE(k + 4); + ctx->input[TR[3]] = LOAD32_LE(k + 8); + ctx->input[TR[4]] = LOAD32_LE(k + 12); + ctx->input[TR[11]] = LOAD32_LE(k + 16); + ctx->input[TR[12]] = LOAD32_LE(k + 20); + ctx->input[TR[13]] = LOAD32_LE(k + 24); + ctx->input[TR[14]] = LOAD32_LE(k + 28); + ctx->input[TR[0]] = 0x61707865; + ctx->input[TR[5]] = 0x3320646e; + ctx->input[TR[10]] = 0x79622d32; + ctx->input[TR[15]] = 0x6b206574; +} + +static void +salsa_ivsetup(salsa_ctx *ctx, const uint8_t *iv, const uint8_t *counter) +{ + ctx->input[TR[6]] = LOAD32_LE(iv + 0); + ctx->input[TR[7]] = LOAD32_LE(iv + 4); + ctx->input[TR[8]] = counter == NULL ? 0 : LOAD32_LE(counter + 0); + ctx->input[TR[9]] = counter == NULL ? 0 : LOAD32_LE(counter + 4); +} + +static void +salsa20_encrypt_bytes(salsa_ctx *ctx, const uint8_t *m, uint8_t *c, + unsigned long long bytes) +{ + uint32_t * const x = &ctx->input[0]; + + if (!bytes) { + return; /* LCOV_EXCL_LINE */ + } + +#include "sodium_xmm6int_u8.h" +#include "sodium_xmm6int_u4.h" +#include "sodium_xmm6int_u1.h" +#include "sodium_xmm6int_u0.h" +} + +static int +stream_avx2(unsigned char *c, unsigned long long clen, const unsigned char *n, + const unsigned char *k) +{ + struct salsa_ctx ctx; + + if (!clen) { + return 0; + } + COMPILER_ASSERT(crypto_stream_salsa20_KEYBYTES == 256 / 8); + salsa_keysetup(&ctx, k); + salsa_ivsetup(&ctx, n, NULL); + memset(c, 0, clen); + salsa20_encrypt_bytes(&ctx, c, c, clen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +static int +stream_avx2_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, uint64_t ic, + const unsigned char *k) +{ + struct salsa_ctx ctx; + uint8_t ic_bytes[8]; + uint32_t ic_high; + uint32_t ic_low; + + if (!mlen) { + return 0; + } + ic_high = (uint32_t) (ic >> 32); + ic_low = (uint32_t) ic; + STORE32_LE(&ic_bytes[0], ic_low); + STORE32_LE(&ic_bytes[4], ic_high); + salsa_keysetup(&ctx, k); + salsa_ivsetup(&ctx, n, ic_bytes); + salsa20_encrypt_bytes(&ctx, m, c, mlen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +struct crypto_stream_salsa20_implementation + crypto_stream_salsa20_xmm6int_avx2_implementation = { + SODIUM_C99(.stream =) stream_avx2, + SODIUM_C99(.stream_xor_ic =) stream_avx2_xor_ic + }; + +#endif + +int salsa2_xmm6int_link_warning_dummy = 0; diff --git a/sodium/sodium_xmm6int_salsa20-avx2.h b/sodium/sodium_xmm6int_salsa20-avx2.h new file mode 100644 index 00000000..1d769304 --- /dev/null +++ b/sodium/sodium_xmm6int_salsa20-avx2.h @@ -0,0 +1,8 @@ + +#include + +#include "sodium_stream_salsa20.h" +#include "sodium_crypto_stream_salsa20.h" + +extern struct crypto_stream_salsa20_implementation + crypto_stream_salsa20_xmm6int_avx2_implementation; diff --git a/sodium/sodium_xmm6int_salsa20-sse2.c b/sodium/sodium_xmm6int_salsa20-sse2.c new file mode 100644 index 00000000..aa56d5a4 --- /dev/null +++ b/sodium/sodium_xmm6int_salsa20-sse2.c @@ -0,0 +1,124 @@ + +#include +#include +#include + +#include "sodium_crypto_stream_salsa20.h" +#include "sodium_private_common.h" +#include "sodium_private_sse2_64_32.h" +#include "sodium_utils.h" + +#ifdef HAVE_EMMINTRIN_H + +# ifdef __GNUC__ +# pragma GCC target("sse2") +# endif +# include + +# include "sodium_stream_salsa20.h" +# include "sodium_xmm6int_salsa20-sse2.h" + +# define ROUNDS 20 + +typedef struct salsa_ctx { + uint32_t input[16]; +} salsa_ctx; + +static const int TR[16] = { + 0, 5, 10, 15, 12, 1, 6, 11, 8, 13, 2, 7, 4, 9, 14, 3 +}; + +static void +salsa_keysetup(salsa_ctx *ctx, const uint8_t *k) +{ + ctx->input[TR[1]] = LOAD32_LE(k + 0); + ctx->input[TR[2]] = LOAD32_LE(k + 4); + ctx->input[TR[3]] = LOAD32_LE(k + 8); + ctx->input[TR[4]] = LOAD32_LE(k + 12); + ctx->input[TR[11]] = LOAD32_LE(k + 16); + ctx->input[TR[12]] = LOAD32_LE(k + 20); + ctx->input[TR[13]] = LOAD32_LE(k + 24); + ctx->input[TR[14]] = LOAD32_LE(k + 28); + ctx->input[TR[0]] = 0x61707865; + ctx->input[TR[5]] = 0x3320646e; + ctx->input[TR[10]] = 0x79622d32; + ctx->input[TR[15]] = 0x6b206574; +} + +static void +salsa_ivsetup(salsa_ctx *ctx, const uint8_t *iv, const uint8_t *counter) +{ + ctx->input[TR[6]] = LOAD32_LE(iv + 0); + ctx->input[TR[7]] = LOAD32_LE(iv + 4); + ctx->input[TR[8]] = counter == NULL ? 0 : LOAD32_LE(counter + 0); + ctx->input[TR[9]] = counter == NULL ? 0 : LOAD32_LE(counter + 4); +} + +static void +salsa20_encrypt_bytes(salsa_ctx *ctx, const uint8_t *m, uint8_t *c, + unsigned long long bytes) +{ + uint32_t * const x = &ctx->input[0]; + + if (!bytes) { + return; /* LCOV_EXCL_LINE */ + } + +#include "sodium_xmm6int_u4.h" +#include "sodium_xmm6int_u1.h" +#include "sodium_xmm6int_u0.h" +} + +static int +stream_sse2(unsigned char *c, unsigned long long clen, const unsigned char *n, + const unsigned char *k) +{ + struct salsa_ctx ctx; + + if (!clen) { + return 0; + } + COMPILER_ASSERT(crypto_stream_salsa20_KEYBYTES == 256 / 8); + salsa_keysetup(&ctx, k); + salsa_ivsetup(&ctx, n, NULL); + memset(c, 0, clen); + salsa20_encrypt_bytes(&ctx, c, c, clen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +static int +stream_sse2_xor_ic(unsigned char *c, const unsigned char *m, + unsigned long long mlen, const unsigned char *n, uint64_t ic, + const unsigned char *k) +{ + struct salsa_ctx ctx; + uint8_t ic_bytes[8]; + uint32_t ic_high; + uint32_t ic_low; + + if (!mlen) { + return 0; + } + ic_high = (uint32_t) (ic >> 32); + ic_low = (uint32_t) (ic); + STORE32_LE(&ic_bytes[0], ic_low); + STORE32_LE(&ic_bytes[4], ic_high); + salsa_keysetup(&ctx, k); + salsa_ivsetup(&ctx, n, ic_bytes); + salsa20_encrypt_bytes(&ctx, m, c, mlen); + sodium_memzero(&ctx, sizeof ctx); + + return 0; +} + +struct crypto_stream_salsa20_implementation + crypto_stream_salsa20_xmm6int_sse2_implementation = { + SODIUM_C99(.stream =) stream_sse2, + SODIUM_C99(.stream_xor_ic =) stream_sse2_xor_ic + }; + +#endif + +int salsa20_xmm6int_sse2_link_warning_dummy = 0; diff --git a/sodium/sodium_xmm6int_salsa20-sse2.h b/sodium/sodium_xmm6int_salsa20-sse2.h new file mode 100644 index 00000000..7c2dce4e --- /dev/null +++ b/sodium/sodium_xmm6int_salsa20-sse2.h @@ -0,0 +1,8 @@ + +#include + +#include "sodium_stream_salsa20.h" +#include "sodium_crypto_stream_salsa20.h" + +extern struct crypto_stream_salsa20_implementation + crypto_stream_salsa20_xmm6int_sse2_implementation; diff --git a/sodium/sodium_xmm6int_u0.h b/sodium/sodium_xmm6int_u0.h new file mode 100644 index 00000000..4262f992 --- /dev/null +++ b/sodium/sodium_xmm6int_u0.h @@ -0,0 +1,195 @@ +if (bytes > 0) { + __m128i diag0 = _mm_loadu_si128((const __m128i *) (x + 0)); + __m128i diag1 = _mm_loadu_si128((const __m128i *) (x + 4)); + __m128i diag2 = _mm_loadu_si128((const __m128i *) (x + 8)); + __m128i diag3 = _mm_loadu_si128((const __m128i *) (x + 12)); + __m128i a0, a1, a2, a3, a4, a5, a6, a7; + __m128i b0, b1, b2, b3, b4, b5, b6, b7; + uint8_t partialblock[64]; + + unsigned int i; + + a0 = diag1; + for (i = 0; i < ROUNDS; i += 4) { + a0 = _mm_add_epi32(a0, diag0); + a1 = diag0; + b0 = a0; + a0 = _mm_slli_epi32(a0, 7); + b0 = _mm_srli_epi32(b0, 25); + diag3 = _mm_xor_si128(diag3, a0); + + diag3 = _mm_xor_si128(diag3, b0); + + a1 = _mm_add_epi32(a1, diag3); + a2 = diag3; + b1 = a1; + a1 = _mm_slli_epi32(a1, 9); + b1 = _mm_srli_epi32(b1, 23); + diag2 = _mm_xor_si128(diag2, a1); + diag3 = _mm_shuffle_epi32(diag3, 0x93); + diag2 = _mm_xor_si128(diag2, b1); + + a2 = _mm_add_epi32(a2, diag2); + a3 = diag2; + b2 = a2; + a2 = _mm_slli_epi32(a2, 13); + b2 = _mm_srli_epi32(b2, 19); + diag1 = _mm_xor_si128(diag1, a2); + diag2 = _mm_shuffle_epi32(diag2, 0x4e); + diag1 = _mm_xor_si128(diag1, b2); + + a3 = _mm_add_epi32(a3, diag1); + a4 = diag3; + b3 = a3; + a3 = _mm_slli_epi32(a3, 18); + b3 = _mm_srli_epi32(b3, 14); + diag0 = _mm_xor_si128(diag0, a3); + diag1 = _mm_shuffle_epi32(diag1, 0x39); + diag0 = _mm_xor_si128(diag0, b3); + + a4 = _mm_add_epi32(a4, diag0); + a5 = diag0; + b4 = a4; + a4 = _mm_slli_epi32(a4, 7); + b4 = _mm_srli_epi32(b4, 25); + diag1 = _mm_xor_si128(diag1, a4); + + diag1 = _mm_xor_si128(diag1, b4); + + a5 = _mm_add_epi32(a5, diag1); + a6 = diag1; + b5 = a5; + a5 = _mm_slli_epi32(a5, 9); + b5 = _mm_srli_epi32(b5, 23); + diag2 = _mm_xor_si128(diag2, a5); + diag1 = _mm_shuffle_epi32(diag1, 0x93); + diag2 = _mm_xor_si128(diag2, b5); + + a6 = _mm_add_epi32(a6, diag2); + a7 = diag2; + b6 = a6; + a6 = _mm_slli_epi32(a6, 13); + b6 = _mm_srli_epi32(b6, 19); + diag3 = _mm_xor_si128(diag3, a6); + diag2 = _mm_shuffle_epi32(diag2, 0x4e); + diag3 = _mm_xor_si128(diag3, b6); + + a7 = _mm_add_epi32(a7, diag3); + a0 = diag1; + b7 = a7; + a7 = _mm_slli_epi32(a7, 18); + b7 = _mm_srli_epi32(b7, 14); + diag0 = _mm_xor_si128(diag0, a7); + diag3 = _mm_shuffle_epi32(diag3, 0x39); + diag0 = _mm_xor_si128(diag0, b7); + + a0 = _mm_add_epi32(a0, diag0); + a1 = diag0; + b0 = a0; + a0 = _mm_slli_epi32(a0, 7); + b0 = _mm_srli_epi32(b0, 25); + diag3 = _mm_xor_si128(diag3, a0); + + diag3 = _mm_xor_si128(diag3, b0); + + a1 = _mm_add_epi32(a1, diag3); + a2 = diag3; + b1 = a1; + a1 = _mm_slli_epi32(a1, 9); + b1 = _mm_srli_epi32(b1, 23); + diag2 = _mm_xor_si128(diag2, a1); + diag3 = _mm_shuffle_epi32(diag3, 0x93); + diag2 = _mm_xor_si128(diag2, b1); + + a2 = _mm_add_epi32(a2, diag2); + a3 = diag2; + b2 = a2; + a2 = _mm_slli_epi32(a2, 13); + b2 = _mm_srli_epi32(b2, 19); + diag1 = _mm_xor_si128(diag1, a2); + diag2 = _mm_shuffle_epi32(diag2, 0x4e); + diag1 = _mm_xor_si128(diag1, b2); + + a3 = _mm_add_epi32(a3, diag1); + a4 = diag3; + b3 = a3; + a3 = _mm_slli_epi32(a3, 18); + b3 = _mm_srli_epi32(b3, 14); + diag0 = _mm_xor_si128(diag0, a3); + diag1 = _mm_shuffle_epi32(diag1, 0x39); + diag0 = _mm_xor_si128(diag0, b3); + + a4 = _mm_add_epi32(a4, diag0); + a5 = diag0; + b4 = a4; + a4 = _mm_slli_epi32(a4, 7); + b4 = _mm_srli_epi32(b4, 25); + diag1 = _mm_xor_si128(diag1, a4); + + diag1 = _mm_xor_si128(diag1, b4); + + a5 = _mm_add_epi32(a5, diag1); + a6 = diag1; + b5 = a5; + a5 = _mm_slli_epi32(a5, 9); + b5 = _mm_srli_epi32(b5, 23); + diag2 = _mm_xor_si128(diag2, a5); + diag1 = _mm_shuffle_epi32(diag1, 0x93); + diag2 = _mm_xor_si128(diag2, b5); + + a6 = _mm_add_epi32(a6, diag2); + a7 = diag2; + b6 = a6; + a6 = _mm_slli_epi32(a6, 13); + b6 = _mm_srli_epi32(b6, 19); + diag3 = _mm_xor_si128(diag3, a6); + diag2 = _mm_shuffle_epi32(diag2, 0x4e); + diag3 = _mm_xor_si128(diag3, b6); + + a7 = _mm_add_epi32(a7, diag3); + a0 = diag1; + b7 = a7; + a7 = _mm_slli_epi32(a7, 18); + b7 = _mm_srli_epi32(b7, 14); + diag0 = _mm_xor_si128(diag0, a7); + diag3 = _mm_shuffle_epi32(diag3, 0x39); + diag0 = _mm_xor_si128(diag0, b7); + } + + diag0 = _mm_add_epi32(diag0, _mm_loadu_si128((const __m128i *) (x + 0))); + diag1 = _mm_add_epi32(diag1, _mm_loadu_si128((const __m128i *) (x + 4))); + diag2 = _mm_add_epi32(diag2, _mm_loadu_si128((const __m128i *) (x + 8))); + diag3 = _mm_add_epi32(diag3, _mm_loadu_si128((const __m128i *) (x + 12))); + +#define ONEQUAD_SHUFFLE(A, B, C, D) \ + do { \ + uint32_t in##A = _mm_cvtsi128_si32(diag0); \ + uint32_t in##B = _mm_cvtsi128_si32(diag1); \ + uint32_t in##C = _mm_cvtsi128_si32(diag2); \ + uint32_t in##D = _mm_cvtsi128_si32(diag3); \ + diag0 = _mm_shuffle_epi32(diag0, 0x39); \ + diag1 = _mm_shuffle_epi32(diag1, 0x39); \ + diag2 = _mm_shuffle_epi32(diag2, 0x39); \ + diag3 = _mm_shuffle_epi32(diag3, 0x39); \ + memcpy( partialblock + (A*4), &in##A, 4 ); \ + memcpy( partialblock + (B*4), &in##B, 4 ); \ + memcpy( partialblock + (C*4), &in##C, 4 ); \ + memcpy( partialblock + (D*4), &in##D, 4 ); \ + } while (0) + +#define ONEQUAD(A, B, C, D) ONEQUAD_SHUFFLE(A, B, C, D) + + ONEQUAD(0, 12, 8, 4); + ONEQUAD(5, 1, 13, 9); + ONEQUAD(10, 6, 2, 14); + ONEQUAD(15, 11, 7, 3); + +#undef ONEQUAD +#undef ONEQUAD_SHUFFLE + + for (i = 0; i < bytes; i++) { + c[i] = m[i] ^ partialblock[i]; + } + + sodium_memzero(partialblock, sizeof partialblock); +} diff --git a/sodium/sodium_xmm6int_u1.h b/sodium/sodium_xmm6int_u1.h new file mode 100644 index 00000000..e82521cd --- /dev/null +++ b/sodium/sodium_xmm6int_u1.h @@ -0,0 +1,207 @@ +while (bytes >= 64) { + __m128i diag0 = _mm_loadu_si128((const __m128i *) (x + 0)); + __m128i diag1 = _mm_loadu_si128((const __m128i *) (x + 4)); + __m128i diag2 = _mm_loadu_si128((const __m128i *) (x + 8)); + __m128i diag3 = _mm_loadu_si128((const __m128i *) (x + 12)); + __m128i a0, a1, a2, a3, a4, a5, a6, a7; + __m128i b0, b1, b2, b3, b4, b5, b6, b7; + + uint32_t in8; + uint32_t in9; + int i; + + a0 = diag1; + for (i = 0; i < ROUNDS; i += 4) { + a0 = _mm_add_epi32(a0, diag0); + a1 = diag0; + b0 = a0; + a0 = _mm_slli_epi32(a0, 7); + b0 = _mm_srli_epi32(b0, 25); + diag3 = _mm_xor_si128(diag3, a0); + + diag3 = _mm_xor_si128(diag3, b0); + + a1 = _mm_add_epi32(a1, diag3); + a2 = diag3; + b1 = a1; + a1 = _mm_slli_epi32(a1, 9); + b1 = _mm_srli_epi32(b1, 23); + diag2 = _mm_xor_si128(diag2, a1); + diag3 = _mm_shuffle_epi32(diag3, 0x93); + diag2 = _mm_xor_si128(diag2, b1); + + a2 = _mm_add_epi32(a2, diag2); + a3 = diag2; + b2 = a2; + a2 = _mm_slli_epi32(a2, 13); + b2 = _mm_srli_epi32(b2, 19); + diag1 = _mm_xor_si128(diag1, a2); + diag2 = _mm_shuffle_epi32(diag2, 0x4e); + diag1 = _mm_xor_si128(diag1, b2); + + a3 = _mm_add_epi32(a3, diag1); + a4 = diag3; + b3 = a3; + a3 = _mm_slli_epi32(a3, 18); + b3 = _mm_srli_epi32(b3, 14); + diag0 = _mm_xor_si128(diag0, a3); + diag1 = _mm_shuffle_epi32(diag1, 0x39); + diag0 = _mm_xor_si128(diag0, b3); + + a4 = _mm_add_epi32(a4, diag0); + a5 = diag0; + b4 = a4; + a4 = _mm_slli_epi32(a4, 7); + b4 = _mm_srli_epi32(b4, 25); + diag1 = _mm_xor_si128(diag1, a4); + + diag1 = _mm_xor_si128(diag1, b4); + + a5 = _mm_add_epi32(a5, diag1); + a6 = diag1; + b5 = a5; + a5 = _mm_slli_epi32(a5, 9); + b5 = _mm_srli_epi32(b5, 23); + diag2 = _mm_xor_si128(diag2, a5); + diag1 = _mm_shuffle_epi32(diag1, 0x93); + diag2 = _mm_xor_si128(diag2, b5); + + a6 = _mm_add_epi32(a6, diag2); + a7 = diag2; + b6 = a6; + a6 = _mm_slli_epi32(a6, 13); + b6 = _mm_srli_epi32(b6, 19); + diag3 = _mm_xor_si128(diag3, a6); + diag2 = _mm_shuffle_epi32(diag2, 0x4e); + diag3 = _mm_xor_si128(diag3, b6); + + a7 = _mm_add_epi32(a7, diag3); + a0 = diag1; + b7 = a7; + a7 = _mm_slli_epi32(a7, 18); + b7 = _mm_srli_epi32(b7, 14); + diag0 = _mm_xor_si128(diag0, a7); + diag3 = _mm_shuffle_epi32(diag3, 0x39); + diag0 = _mm_xor_si128(diag0, b7); + + a0 = _mm_add_epi32(a0, diag0); + a1 = diag0; + b0 = a0; + a0 = _mm_slli_epi32(a0, 7); + b0 = _mm_srli_epi32(b0, 25); + diag3 = _mm_xor_si128(diag3, a0); + + diag3 = _mm_xor_si128(diag3, b0); + + a1 = _mm_add_epi32(a1, diag3); + a2 = diag3; + b1 = a1; + a1 = _mm_slli_epi32(a1, 9); + b1 = _mm_srli_epi32(b1, 23); + diag2 = _mm_xor_si128(diag2, a1); + diag3 = _mm_shuffle_epi32(diag3, 0x93); + diag2 = _mm_xor_si128(diag2, b1); + + a2 = _mm_add_epi32(a2, diag2); + a3 = diag2; + b2 = a2; + a2 = _mm_slli_epi32(a2, 13); + b2 = _mm_srli_epi32(b2, 19); + diag1 = _mm_xor_si128(diag1, a2); + diag2 = _mm_shuffle_epi32(diag2, 0x4e); + diag1 = _mm_xor_si128(diag1, b2); + + a3 = _mm_add_epi32(a3, diag1); + a4 = diag3; + b3 = a3; + a3 = _mm_slli_epi32(a3, 18); + b3 = _mm_srli_epi32(b3, 14); + diag0 = _mm_xor_si128(diag0, a3); + diag1 = _mm_shuffle_epi32(diag1, 0x39); + diag0 = _mm_xor_si128(diag0, b3); + + a4 = _mm_add_epi32(a4, diag0); + a5 = diag0; + b4 = a4; + a4 = _mm_slli_epi32(a4, 7); + b4 = _mm_srli_epi32(b4, 25); + diag1 = _mm_xor_si128(diag1, a4); + + diag1 = _mm_xor_si128(diag1, b4); + + a5 = _mm_add_epi32(a5, diag1); + a6 = diag1; + b5 = a5; + a5 = _mm_slli_epi32(a5, 9); + b5 = _mm_srli_epi32(b5, 23); + diag2 = _mm_xor_si128(diag2, a5); + diag1 = _mm_shuffle_epi32(diag1, 0x93); + diag2 = _mm_xor_si128(diag2, b5); + + a6 = _mm_add_epi32(a6, diag2); + a7 = diag2; + b6 = a6; + a6 = _mm_slli_epi32(a6, 13); + b6 = _mm_srli_epi32(b6, 19); + diag3 = _mm_xor_si128(diag3, a6); + diag2 = _mm_shuffle_epi32(diag2, 0x4e); + diag3 = _mm_xor_si128(diag3, b6); + + a7 = _mm_add_epi32(a7, diag3); + a0 = diag1; + b7 = a7; + a7 = _mm_slli_epi32(a7, 18); + b7 = _mm_srli_epi32(b7, 14); + diag0 = _mm_xor_si128(diag0, a7); + diag3 = _mm_shuffle_epi32(diag3, 0x39); + diag0 = _mm_xor_si128(diag0, b7); + } + + diag0 = _mm_add_epi32(diag0, _mm_loadu_si128((const __m128i *) (x + 0))); + diag1 = _mm_add_epi32(diag1, _mm_loadu_si128((const __m128i *) (x + 4))); + diag2 = _mm_add_epi32(diag2, _mm_loadu_si128((const __m128i *) (x + 8))); + diag3 = _mm_add_epi32(diag3, _mm_loadu_si128((const __m128i *) (x + 12))); + +#define ONEQUAD_SHUFFLE(A, B, C, D) \ + do { \ + uint32_t in##A = _mm_cvtsi128_si32(diag0); \ + uint32_t in##B = _mm_cvtsi128_si32(diag1); \ + uint32_t in##C = _mm_cvtsi128_si32(diag2); \ + uint32_t in##D = _mm_cvtsi128_si32(diag3); \ + diag0 = _mm_shuffle_epi32(diag0, 0x39); \ + diag1 = _mm_shuffle_epi32(diag1, 0x39); \ + diag2 = _mm_shuffle_epi32(diag2, 0x39); \ + diag3 = _mm_shuffle_epi32(diag3, 0x39); \ + in##A ^= *(const uint32_t *) (m + (A * 4)); \ + in##B ^= *(const uint32_t *) (m + (B * 4)); \ + in##C ^= *(const uint32_t *) (m + (C * 4)); \ + in##D ^= *(const uint32_t *) (m + (D * 4)); \ + *(uint32_t *) (c + (A * 4)) = in##A; \ + *(uint32_t *) (c + (B * 4)) = in##B; \ + *(uint32_t *) (c + (C * 4)) = in##C; \ + *(uint32_t *) (c + (D * 4)) = in##D; \ + } while (0) + +#define ONEQUAD(A, B, C, D) ONEQUAD_SHUFFLE(A, B, C, D) + + ONEQUAD(0, 12, 8, 4); + ONEQUAD(5, 1, 13, 9); + ONEQUAD(10, 6, 2, 14); + ONEQUAD(15, 11, 7, 3); + +#undef ONEQUAD +#undef ONEQUAD_SHUFFLE + + in8 = x[8]; + in9 = x[13]; + in8++; + if (in8 == 0) { + in9++; + } + x[8] = in8; + x[13] = in9; + + c += 64; + m += 64; + bytes -= 64; +} diff --git a/sodium/sodium_xmm6int_u4.h b/sodium/sodium_xmm6int_u4.h new file mode 100644 index 00000000..474f4860 --- /dev/null +++ b/sodium/sodium_xmm6int_u4.h @@ -0,0 +1,547 @@ +if (bytes >= 256) { + __m128i y0, y1, y2, y3, y4, y5, y6, y7, y8, y9, y10, y11, y12, y13, y14, + y15; + __m128i z0, z1, z2, z3, z4, z5, z6, z7, z8, z9, z10, z11, z12, z13, z14, + z15; + __m128i orig0, orig1, orig2, orig3, orig4, orig5, orig6, orig7, orig8, + orig9, orig10, orig11, orig12, orig13, orig14, orig15; + + uint32_t in8; + uint32_t in9; + int i; + + /* element broadcast immediate for _mm_shuffle_epi32 are in order: + 0x00, 0x55, 0xaa, 0xff */ + z0 = _mm_loadu_si128((const __m128i *) (x + 0)); + z5 = _mm_shuffle_epi32(z0, 0x55); + z10 = _mm_shuffle_epi32(z0, 0xaa); + z15 = _mm_shuffle_epi32(z0, 0xff); + z0 = _mm_shuffle_epi32(z0, 0x00); + z1 = _mm_loadu_si128((const __m128i *) (x + 4)); + z6 = _mm_shuffle_epi32(z1, 0xaa); + z11 = _mm_shuffle_epi32(z1, 0xff); + z12 = _mm_shuffle_epi32(z1, 0x00); + z1 = _mm_shuffle_epi32(z1, 0x55); + z2 = _mm_loadu_si128((const __m128i *) (x + 8)); + z7 = _mm_shuffle_epi32(z2, 0xff); + z13 = _mm_shuffle_epi32(z2, 0x55); + z2 = _mm_shuffle_epi32(z2, 0xaa); + /* no z8 -> first half of the nonce, will fill later */ + z3 = _mm_loadu_si128((const __m128i *) (x + 12)); + z4 = _mm_shuffle_epi32(z3, 0x00); + z14 = _mm_shuffle_epi32(z3, 0xaa); + z3 = _mm_shuffle_epi32(z3, 0xff); + /* no z9 -> second half of the nonce, will fill later */ + orig0 = z0; + orig1 = z1; + orig2 = z2; + orig3 = z3; + orig4 = z4; + orig5 = z5; + orig6 = z6; + orig7 = z7; + orig10 = z10; + orig11 = z11; + orig12 = z12; + orig13 = z13; + orig14 = z14; + orig15 = z15; + + while (bytes >= 256) { + /* vector implementation for z8 and z9 */ + /* not sure if it helps for only 4 blocks */ + const __m128i addv8 = _mm_set_epi64x(1, 0); + const __m128i addv9 = _mm_set_epi64x(3, 2); + __m128i t8, t9; + uint64_t in89; + + in8 = x[8]; + in9 = x[13]; + in89 = ((uint64_t) in8) | (((uint64_t) in9) << 32); + t8 = _mm_set1_epi64x(in89); + t9 = _mm_set1_epi64x(in89); + + z8 = _mm_add_epi64(addv8, t8); + z9 = _mm_add_epi64(addv9, t9); + + t8 = _mm_unpacklo_epi32(z8, z9); + t9 = _mm_unpackhi_epi32(z8, z9); + + z8 = _mm_unpacklo_epi32(t8, t9); + z9 = _mm_unpackhi_epi32(t8, t9); + + orig8 = z8; + orig9 = z9; + + in89 += 4; + + x[8] = in89 & 0xFFFFFFFF; + x[13] = (in89 >> 32) & 0xFFFFFFFF; + + z5 = orig5; + z10 = orig10; + z15 = orig15; + z14 = orig14; + z3 = orig3; + z6 = orig6; + z11 = orig11; + z1 = orig1; + + z7 = orig7; + z13 = orig13; + z2 = orig2; + z9 = orig9; + z0 = orig0; + z12 = orig12; + z4 = orig4; + z8 = orig8; + + for (i = 0; i < ROUNDS; i += 2) { + /* the inner loop is a direct translation (regexp search/replace) + * from the amd64-xmm6 ASM */ + __m128i r0, r1, r2, r3, r4, r5, r6, r7, r8, r9, r10, r11, r12, r13, + r14, r15; + + y4 = z12; + y4 = _mm_add_epi32(y4, z0); + r4 = y4; + y4 = _mm_slli_epi32(y4, 7); + z4 = _mm_xor_si128(z4, y4); + r4 = _mm_srli_epi32(r4, 25); + z4 = _mm_xor_si128(z4, r4); + + y9 = z1; + y9 = _mm_add_epi32(y9, z5); + r9 = y9; + y9 = _mm_slli_epi32(y9, 7); + z9 = _mm_xor_si128(z9, y9); + r9 = _mm_srli_epi32(r9, 25); + z9 = _mm_xor_si128(z9, r9); + + y8 = z0; + y8 = _mm_add_epi32(y8, z4); + r8 = y8; + y8 = _mm_slli_epi32(y8, 9); + z8 = _mm_xor_si128(z8, y8); + r8 = _mm_srli_epi32(r8, 23); + z8 = _mm_xor_si128(z8, r8); + + y13 = z5; + y13 = _mm_add_epi32(y13, z9); + r13 = y13; + y13 = _mm_slli_epi32(y13, 9); + z13 = _mm_xor_si128(z13, y13); + r13 = _mm_srli_epi32(r13, 23); + z13 = _mm_xor_si128(z13, r13); + + y12 = z4; + y12 = _mm_add_epi32(y12, z8); + r12 = y12; + y12 = _mm_slli_epi32(y12, 13); + z12 = _mm_xor_si128(z12, y12); + r12 = _mm_srli_epi32(r12, 19); + z12 = _mm_xor_si128(z12, r12); + + y1 = z9; + y1 = _mm_add_epi32(y1, z13); + r1 = y1; + y1 = _mm_slli_epi32(y1, 13); + z1 = _mm_xor_si128(z1, y1); + r1 = _mm_srli_epi32(r1, 19); + z1 = _mm_xor_si128(z1, r1); + + y0 = z8; + y0 = _mm_add_epi32(y0, z12); + r0 = y0; + y0 = _mm_slli_epi32(y0, 18); + z0 = _mm_xor_si128(z0, y0); + r0 = _mm_srli_epi32(r0, 14); + z0 = _mm_xor_si128(z0, r0); + + y5 = z13; + y5 = _mm_add_epi32(y5, z1); + r5 = y5; + y5 = _mm_slli_epi32(y5, 18); + z5 = _mm_xor_si128(z5, y5); + r5 = _mm_srli_epi32(r5, 14); + z5 = _mm_xor_si128(z5, r5); + + y14 = z6; + y14 = _mm_add_epi32(y14, z10); + r14 = y14; + y14 = _mm_slli_epi32(y14, 7); + z14 = _mm_xor_si128(z14, y14); + r14 = _mm_srli_epi32(r14, 25); + z14 = _mm_xor_si128(z14, r14); + + y3 = z11; + y3 = _mm_add_epi32(y3, z15); + r3 = y3; + y3 = _mm_slli_epi32(y3, 7); + z3 = _mm_xor_si128(z3, y3); + r3 = _mm_srli_epi32(r3, 25); + z3 = _mm_xor_si128(z3, r3); + + y2 = z10; + y2 = _mm_add_epi32(y2, z14); + r2 = y2; + y2 = _mm_slli_epi32(y2, 9); + z2 = _mm_xor_si128(z2, y2); + r2 = _mm_srli_epi32(r2, 23); + z2 = _mm_xor_si128(z2, r2); + + y7 = z15; + y7 = _mm_add_epi32(y7, z3); + r7 = y7; + y7 = _mm_slli_epi32(y7, 9); + z7 = _mm_xor_si128(z7, y7); + r7 = _mm_srli_epi32(r7, 23); + z7 = _mm_xor_si128(z7, r7); + + y6 = z14; + y6 = _mm_add_epi32(y6, z2); + r6 = y6; + y6 = _mm_slli_epi32(y6, 13); + z6 = _mm_xor_si128(z6, y6); + r6 = _mm_srli_epi32(r6, 19); + z6 = _mm_xor_si128(z6, r6); + + y11 = z3; + y11 = _mm_add_epi32(y11, z7); + r11 = y11; + y11 = _mm_slli_epi32(y11, 13); + z11 = _mm_xor_si128(z11, y11); + r11 = _mm_srli_epi32(r11, 19); + z11 = _mm_xor_si128(z11, r11); + + y10 = z2; + y10 = _mm_add_epi32(y10, z6); + r10 = y10; + y10 = _mm_slli_epi32(y10, 18); + z10 = _mm_xor_si128(z10, y10); + r10 = _mm_srli_epi32(r10, 14); + z10 = _mm_xor_si128(z10, r10); + + y1 = z3; + y1 = _mm_add_epi32(y1, z0); + r1 = y1; + y1 = _mm_slli_epi32(y1, 7); + z1 = _mm_xor_si128(z1, y1); + r1 = _mm_srli_epi32(r1, 25); + z1 = _mm_xor_si128(z1, r1); + + y15 = z7; + y15 = _mm_add_epi32(y15, z11); + r15 = y15; + y15 = _mm_slli_epi32(y15, 18); + z15 = _mm_xor_si128(z15, y15); + r15 = _mm_srli_epi32(r15, 14); + z15 = _mm_xor_si128(z15, r15); + + y6 = z4; + y6 = _mm_add_epi32(y6, z5); + r6 = y6; + y6 = _mm_slli_epi32(y6, 7); + z6 = _mm_xor_si128(z6, y6); + r6 = _mm_srli_epi32(r6, 25); + z6 = _mm_xor_si128(z6, r6); + + y2 = z0; + y2 = _mm_add_epi32(y2, z1); + r2 = y2; + y2 = _mm_slli_epi32(y2, 9); + z2 = _mm_xor_si128(z2, y2); + r2 = _mm_srli_epi32(r2, 23); + z2 = _mm_xor_si128(z2, r2); + + y7 = z5; + y7 = _mm_add_epi32(y7, z6); + r7 = y7; + y7 = _mm_slli_epi32(y7, 9); + z7 = _mm_xor_si128(z7, y7); + r7 = _mm_srli_epi32(r7, 23); + z7 = _mm_xor_si128(z7, r7); + + y3 = z1; + y3 = _mm_add_epi32(y3, z2); + r3 = y3; + y3 = _mm_slli_epi32(y3, 13); + z3 = _mm_xor_si128(z3, y3); + r3 = _mm_srli_epi32(r3, 19); + z3 = _mm_xor_si128(z3, r3); + + y4 = z6; + y4 = _mm_add_epi32(y4, z7); + r4 = y4; + y4 = _mm_slli_epi32(y4, 13); + z4 = _mm_xor_si128(z4, y4); + r4 = _mm_srli_epi32(r4, 19); + z4 = _mm_xor_si128(z4, r4); + + y0 = z2; + y0 = _mm_add_epi32(y0, z3); + r0 = y0; + y0 = _mm_slli_epi32(y0, 18); + z0 = _mm_xor_si128(z0, y0); + r0 = _mm_srli_epi32(r0, 14); + z0 = _mm_xor_si128(z0, r0); + + y5 = z7; + y5 = _mm_add_epi32(y5, z4); + r5 = y5; + y5 = _mm_slli_epi32(y5, 18); + z5 = _mm_xor_si128(z5, y5); + r5 = _mm_srli_epi32(r5, 14); + z5 = _mm_xor_si128(z5, r5); + + y11 = z9; + y11 = _mm_add_epi32(y11, z10); + r11 = y11; + y11 = _mm_slli_epi32(y11, 7); + z11 = _mm_xor_si128(z11, y11); + r11 = _mm_srli_epi32(r11, 25); + z11 = _mm_xor_si128(z11, r11); + + y12 = z14; + y12 = _mm_add_epi32(y12, z15); + r12 = y12; + y12 = _mm_slli_epi32(y12, 7); + z12 = _mm_xor_si128(z12, y12); + r12 = _mm_srli_epi32(r12, 25); + z12 = _mm_xor_si128(z12, r12); + + y8 = z10; + y8 = _mm_add_epi32(y8, z11); + r8 = y8; + y8 = _mm_slli_epi32(y8, 9); + z8 = _mm_xor_si128(z8, y8); + r8 = _mm_srli_epi32(r8, 23); + z8 = _mm_xor_si128(z8, r8); + + y13 = z15; + y13 = _mm_add_epi32(y13, z12); + r13 = y13; + y13 = _mm_slli_epi32(y13, 9); + z13 = _mm_xor_si128(z13, y13); + r13 = _mm_srli_epi32(r13, 23); + z13 = _mm_xor_si128(z13, r13); + + y9 = z11; + y9 = _mm_add_epi32(y9, z8); + r9 = y9; + y9 = _mm_slli_epi32(y9, 13); + z9 = _mm_xor_si128(z9, y9); + r9 = _mm_srli_epi32(r9, 19); + z9 = _mm_xor_si128(z9, r9); + + y14 = z12; + y14 = _mm_add_epi32(y14, z13); + r14 = y14; + y14 = _mm_slli_epi32(y14, 13); + z14 = _mm_xor_si128(z14, y14); + r14 = _mm_srli_epi32(r14, 19); + z14 = _mm_xor_si128(z14, r14); + + y10 = z8; + y10 = _mm_add_epi32(y10, z9); + r10 = y10; + y10 = _mm_slli_epi32(y10, 18); + z10 = _mm_xor_si128(z10, y10); + r10 = _mm_srli_epi32(r10, 14); + z10 = _mm_xor_si128(z10, r10); + + y15 = z13; + y15 = _mm_add_epi32(y15, z14); + r15 = y15; + y15 = _mm_slli_epi32(y15, 18); + z15 = _mm_xor_si128(z15, y15); + r15 = _mm_srli_epi32(r15, 14); + z15 = _mm_xor_si128(z15, r15); + } + +/* store data ; this macro replicates the original amd64-xmm6 code */ +#define ONEQUAD_SHUFFLE(A, B, C, D) \ + z##A = _mm_add_epi32(z##A, orig##A); \ + z##B = _mm_add_epi32(z##B, orig##B); \ + z##C = _mm_add_epi32(z##C, orig##C); \ + z##D = _mm_add_epi32(z##D, orig##D); \ + in##A = _mm_cvtsi128_si32(z##A); \ + in##B = _mm_cvtsi128_si32(z##B); \ + in##C = _mm_cvtsi128_si32(z##C); \ + in##D = _mm_cvtsi128_si32(z##D); \ + z##A = _mm_shuffle_epi32(z##A, 0x39); \ + z##B = _mm_shuffle_epi32(z##B, 0x39); \ + z##C = _mm_shuffle_epi32(z##C, 0x39); \ + z##D = _mm_shuffle_epi32(z##D, 0x39); \ + \ + in##A ^= *(uint32_t *) (m + 0); \ + in##B ^= *(uint32_t *) (m + 4); \ + in##C ^= *(uint32_t *) (m + 8); \ + in##D ^= *(uint32_t *) (m + 12); \ + \ + *(uint32_t *) (c + 0) = in##A; \ + *(uint32_t *) (c + 4) = in##B; \ + *(uint32_t *) (c + 8) = in##C; \ + *(uint32_t *) (c + 12) = in##D; \ + \ + in##A = _mm_cvtsi128_si32(z##A); \ + in##B = _mm_cvtsi128_si32(z##B); \ + in##C = _mm_cvtsi128_si32(z##C); \ + in##D = _mm_cvtsi128_si32(z##D); \ + z##A = _mm_shuffle_epi32(z##A, 0x39); \ + z##B = _mm_shuffle_epi32(z##B, 0x39); \ + z##C = _mm_shuffle_epi32(z##C, 0x39); \ + z##D = _mm_shuffle_epi32(z##D, 0x39); \ + \ + in##A ^= *(uint32_t *) (m + 64); \ + in##B ^= *(uint32_t *) (m + 68); \ + in##C ^= *(uint32_t *) (m + 72); \ + in##D ^= *(uint32_t *) (m + 76); \ + *(uint32_t *) (c + 64) = in##A; \ + *(uint32_t *) (c + 68) = in##B; \ + *(uint32_t *) (c + 72) = in##C; \ + *(uint32_t *) (c + 76) = in##D; \ + \ + in##A = _mm_cvtsi128_si32(z##A); \ + in##B = _mm_cvtsi128_si32(z##B); \ + in##C = _mm_cvtsi128_si32(z##C); \ + in##D = _mm_cvtsi128_si32(z##D); \ + z##A = _mm_shuffle_epi32(z##A, 0x39); \ + z##B = _mm_shuffle_epi32(z##B, 0x39); \ + z##C = _mm_shuffle_epi32(z##C, 0x39); \ + z##D = _mm_shuffle_epi32(z##D, 0x39); \ + \ + in##A ^= *(uint32_t *) (m + 128); \ + in##B ^= *(uint32_t *) (m + 132); \ + in##C ^= *(uint32_t *) (m + 136); \ + in##D ^= *(uint32_t *) (m + 140); \ + *(uint32_t *) (c + 128) = in##A; \ + *(uint32_t *) (c + 132) = in##B; \ + *(uint32_t *) (c + 136) = in##C; \ + *(uint32_t *) (c + 140) = in##D; \ + \ + in##A = _mm_cvtsi128_si32(z##A); \ + in##B = _mm_cvtsi128_si32(z##B); \ + in##C = _mm_cvtsi128_si32(z##C); \ + in##D = _mm_cvtsi128_si32(z##D); \ + \ + in##A ^= *(uint32_t *) (m + 192); \ + in##B ^= *(uint32_t *) (m + 196); \ + in##C ^= *(uint32_t *) (m + 200); \ + in##D ^= *(uint32_t *) (m + 204); \ + *(uint32_t *) (c + 192) = in##A; \ + *(uint32_t *) (c + 196) = in##B; \ + *(uint32_t *) (c + 200) = in##C; \ + *(uint32_t *) (c + 204) = in##D + +/* store data ; this macro replaces shuffle+mov by a direct extract; not much + * difference */ +#define ONEQUAD_EXTRACT(A, B, C, D) \ + z##A = _mm_add_epi32(z##A, orig##A); \ + z##B = _mm_add_epi32(z##B, orig##B); \ + z##C = _mm_add_epi32(z##C, orig##C); \ + z##D = _mm_add_epi32(z##D, orig##D); \ + in##A = _mm_cvtsi128_si32(z##A); \ + in##B = _mm_cvtsi128_si32(z##B); \ + in##C = _mm_cvtsi128_si32(z##C); \ + in##D = _mm_cvtsi128_si32(z##D); \ + in##A ^= *(uint32_t *) (m + 0); \ + in##B ^= *(uint32_t *) (m + 4); \ + in##C ^= *(uint32_t *) (m + 8); \ + in##D ^= *(uint32_t *) (m + 12); \ + *(uint32_t *) (c + 0) = in##A; \ + *(uint32_t *) (c + 4) = in##B; \ + *(uint32_t *) (c + 8) = in##C; \ + *(uint32_t *) (c + 12) = in##D; \ + \ + in##A = _mm_extract_epi32(z##A, 1); \ + in##B = _mm_extract_epi32(z##B, 1); \ + in##C = _mm_extract_epi32(z##C, 1); \ + in##D = _mm_extract_epi32(z##D, 1); \ + \ + in##A ^= *(uint32_t *) (m + 64); \ + in##B ^= *(uint32_t *) (m + 68); \ + in##C ^= *(uint32_t *) (m + 72); \ + in##D ^= *(uint32_t *) (m + 76); \ + *(uint32_t *) (c + 64) = in##A; \ + *(uint32_t *) (c + 68) = in##B; \ + *(uint32_t *) (c + 72) = in##C; \ + *(uint32_t *) (c + 76) = in##D; \ + \ + in##A = _mm_extract_epi32(z##A, 2); \ + in##B = _mm_extract_epi32(z##B, 2); \ + in##C = _mm_extract_epi32(z##C, 2); \ + in##D = _mm_extract_epi32(z##D, 2); \ + \ + in##A ^= *(uint32_t *) (m + 128); \ + in##B ^= *(uint32_t *) (m + 132); \ + in##C ^= *(uint32_t *) (m + 136); \ + in##D ^= *(uint32_t *) (m + 140); \ + *(uint32_t *) (c + 128) = in##A; \ + *(uint32_t *) (c + 132) = in##B; \ + *(uint32_t *) (c + 136) = in##C; \ + *(uint32_t *) (c + 140) = in##D; \ + \ + in##A = _mm_extract_epi32(z##A, 3); \ + in##B = _mm_extract_epi32(z##B, 3); \ + in##C = _mm_extract_epi32(z##C, 3); \ + in##D = _mm_extract_epi32(z##D, 3); \ + \ + in##A ^= *(uint32_t *) (m + 192); \ + in##B ^= *(uint32_t *) (m + 196); \ + in##C ^= *(uint32_t *) (m + 200); \ + in##D ^= *(uint32_t *) (m + 204); \ + *(uint32_t *) (c + 192) = in##A; \ + *(uint32_t *) (c + 196) = in##B; \ + *(uint32_t *) (c + 200) = in##C; \ + *(uint32_t *) (c + 204) = in##D + +/* store data ; this macro first transpose data in-registers, and then store + * them in memory. much faster with icc. */ +#define ONEQUAD_TRANSPOSE(A, B, C, D) \ + z##A = _mm_add_epi32(z##A, orig##A); \ + z##B = _mm_add_epi32(z##B, orig##B); \ + z##C = _mm_add_epi32(z##C, orig##C); \ + z##D = _mm_add_epi32(z##D, orig##D); \ + y##A = _mm_unpacklo_epi32(z##A, z##B); \ + y##B = _mm_unpacklo_epi32(z##C, z##D); \ + y##C = _mm_unpackhi_epi32(z##A, z##B); \ + y##D = _mm_unpackhi_epi32(z##C, z##D); \ + z##A = _mm_unpacklo_epi64(y##A, y##B); \ + z##B = _mm_unpackhi_epi64(y##A, y##B); \ + z##C = _mm_unpacklo_epi64(y##C, y##D); \ + z##D = _mm_unpackhi_epi64(y##C, y##D); \ + y##A = _mm_xor_si128(z##A, _mm_loadu_si128((const __m128i *) (m + 0))); \ + _mm_storeu_si128((__m128i *) (c + 0), y##A); \ + y##B = _mm_xor_si128(z##B, _mm_loadu_si128((const __m128i *) (m + 64))); \ + _mm_storeu_si128((__m128i *) (c + 64), y##B); \ + y##C = _mm_xor_si128(z##C, _mm_loadu_si128((const __m128i *) (m + 128))); \ + _mm_storeu_si128((__m128i *) (c + 128), y##C); \ + y##D = _mm_xor_si128(z##D, _mm_loadu_si128((const __m128i *) (m + 192))); \ + _mm_storeu_si128((__m128i *) (c + 192), y##D) + +#define ONEQUAD(A, B, C, D) ONEQUAD_TRANSPOSE(A, B, C, D) + + ONEQUAD(0, 1, 2, 3); + m += 16; + c += 16; + ONEQUAD(4, 5, 6, 7); + m += 16; + c += 16; + ONEQUAD(8, 9, 10, 11); + m += 16; + c += 16; + ONEQUAD(12, 13, 14, 15); + m -= 48; + c -= 48; + +#undef ONEQUAD +#undef ONEQUAD_TRANSPOSE +#undef ONEQUAD_EXTRACT +#undef ONEQUAD_SHUFFLE + + bytes -= 256; + c += 256; + m += 256; + } +} diff --git a/sodium/sodium_xmm6int_u8.h b/sodium/sodium_xmm6int_u8.h new file mode 100644 index 00000000..581b22c2 --- /dev/null +++ b/sodium/sodium_xmm6int_u8.h @@ -0,0 +1,477 @@ +if (bytes >= 512) { + __m256i y0, y1, y2, y3, y4, y5, y6, y7, y8, y9, y10, y11, y12, y13, y14, + y15; + + /* the naive way seems as fast (if not a bit faster) than the vector way */ + __m256i z0 = _mm256_set1_epi32(x[0]); + __m256i z5 = _mm256_set1_epi32(x[1]); + __m256i z10 = _mm256_set1_epi32(x[2]); + __m256i z15 = _mm256_set1_epi32(x[3]); + __m256i z12 = _mm256_set1_epi32(x[4]); + __m256i z1 = _mm256_set1_epi32(x[5]); + __m256i z6 = _mm256_set1_epi32(x[6]); + __m256i z11 = _mm256_set1_epi32(x[7]); + __m256i z8; /* useless */ + __m256i z13 = _mm256_set1_epi32(x[9]); + __m256i z2 = _mm256_set1_epi32(x[10]); + __m256i z7 = _mm256_set1_epi32(x[11]); + __m256i z4 = _mm256_set1_epi32(x[12]); + __m256i z9; /* useless */ + __m256i z14 = _mm256_set1_epi32(x[14]); + __m256i z3 = _mm256_set1_epi32(x[15]); + + __m256i orig0 = z0; + __m256i orig1 = z1; + __m256i orig2 = z2; + __m256i orig3 = z3; + __m256i orig4 = z4; + __m256i orig5 = z5; + __m256i orig6 = z6; + __m256i orig7 = z7; + __m256i orig8; + __m256i orig9; + __m256i orig10 = z10; + __m256i orig11 = z11; + __m256i orig12 = z12; + __m256i orig13 = z13; + __m256i orig14 = z14; + __m256i orig15 = z15; + + uint32_t in8; + uint32_t in9; + int i; + + while (bytes >= 512) { + /* vector implementation for z8 and z9 */ + /* faster than the naive version for 8 blocks */ + const __m256i addv8 = _mm256_set_epi64x(3, 2, 1, 0); + const __m256i addv9 = _mm256_set_epi64x(7, 6, 5, 4); + const __m256i permute = _mm256_set_epi32(7, 6, 3, 2, 5, 4, 1, 0); + + __m256i t8, t9; + uint64_t in89; + + in8 = x[8]; + in9 = x[13]; /* see arrays above for the address translation */ + in89 = ((uint64_t) in8) | (((uint64_t) in9) << 32); + + z8 = z9 = _mm256_broadcastq_epi64(_mm_cvtsi64_si128(in89)); + + t8 = _mm256_add_epi64(addv8, z8); + t9 = _mm256_add_epi64(addv9, z9); + + z8 = _mm256_unpacklo_epi32(t8, t9); + z9 = _mm256_unpackhi_epi32(t8, t9); + + t8 = _mm256_unpacklo_epi32(z8, z9); + t9 = _mm256_unpackhi_epi32(z8, z9); + + /* required because unpack* are intra-lane */ + z8 = _mm256_permutevar8x32_epi32(t8, permute); + z9 = _mm256_permutevar8x32_epi32(t9, permute); + + orig8 = z8; + orig9 = z9; + + in89 += 8; + + x[8] = in89 & 0xFFFFFFFF; + x[13] = (in89 >> 32) & 0xFFFFFFFF; + + z5 = orig5; + z10 = orig10; + z15 = orig15; + z14 = orig14; + z3 = orig3; + z6 = orig6; + z11 = orig11; + z1 = orig1; + + z7 = orig7; + z13 = orig13; + z2 = orig2; + z9 = orig9; + z0 = orig0; + z12 = orig12; + z4 = orig4; + z8 = orig8; + + for (i = 0; i < ROUNDS; i += 2) { + /* the inner loop is a direct translation (regexp search/replace) + * from the amd64-xmm6 ASM */ + __m256i r0, r1, r2, r3, r4, r5, r6, r7, r8, r9, r10, r11, r12, r13, + r14, r15; + + y4 = z12; + y4 = _mm256_add_epi32(y4, z0); + r4 = y4; + y4 = _mm256_slli_epi32(y4, 7); + z4 = _mm256_xor_si256(z4, y4); + r4 = _mm256_srli_epi32(r4, 25); + z4 = _mm256_xor_si256(z4, r4); + + y9 = z1; + y9 = _mm256_add_epi32(y9, z5); + r9 = y9; + y9 = _mm256_slli_epi32(y9, 7); + z9 = _mm256_xor_si256(z9, y9); + r9 = _mm256_srli_epi32(r9, 25); + z9 = _mm256_xor_si256(z9, r9); + + y8 = z0; + y8 = _mm256_add_epi32(y8, z4); + r8 = y8; + y8 = _mm256_slli_epi32(y8, 9); + z8 = _mm256_xor_si256(z8, y8); + r8 = _mm256_srli_epi32(r8, 23); + z8 = _mm256_xor_si256(z8, r8); + + y13 = z5; + y13 = _mm256_add_epi32(y13, z9); + r13 = y13; + y13 = _mm256_slli_epi32(y13, 9); + z13 = _mm256_xor_si256(z13, y13); + r13 = _mm256_srli_epi32(r13, 23); + z13 = _mm256_xor_si256(z13, r13); + + y12 = z4; + y12 = _mm256_add_epi32(y12, z8); + r12 = y12; + y12 = _mm256_slli_epi32(y12, 13); + z12 = _mm256_xor_si256(z12, y12); + r12 = _mm256_srli_epi32(r12, 19); + z12 = _mm256_xor_si256(z12, r12); + + y1 = z9; + y1 = _mm256_add_epi32(y1, z13); + r1 = y1; + y1 = _mm256_slli_epi32(y1, 13); + z1 = _mm256_xor_si256(z1, y1); + r1 = _mm256_srli_epi32(r1, 19); + z1 = _mm256_xor_si256(z1, r1); + + y0 = z8; + y0 = _mm256_add_epi32(y0, z12); + r0 = y0; + y0 = _mm256_slli_epi32(y0, 18); + z0 = _mm256_xor_si256(z0, y0); + r0 = _mm256_srli_epi32(r0, 14); + z0 = _mm256_xor_si256(z0, r0); + + y5 = z13; + y5 = _mm256_add_epi32(y5, z1); + r5 = y5; + y5 = _mm256_slli_epi32(y5, 18); + z5 = _mm256_xor_si256(z5, y5); + r5 = _mm256_srli_epi32(r5, 14); + z5 = _mm256_xor_si256(z5, r5); + + y14 = z6; + y14 = _mm256_add_epi32(y14, z10); + r14 = y14; + y14 = _mm256_slli_epi32(y14, 7); + z14 = _mm256_xor_si256(z14, y14); + r14 = _mm256_srli_epi32(r14, 25); + z14 = _mm256_xor_si256(z14, r14); + + y3 = z11; + y3 = _mm256_add_epi32(y3, z15); + r3 = y3; + y3 = _mm256_slli_epi32(y3, 7); + z3 = _mm256_xor_si256(z3, y3); + r3 = _mm256_srli_epi32(r3, 25); + z3 = _mm256_xor_si256(z3, r3); + + y2 = z10; + y2 = _mm256_add_epi32(y2, z14); + r2 = y2; + y2 = _mm256_slli_epi32(y2, 9); + z2 = _mm256_xor_si256(z2, y2); + r2 = _mm256_srli_epi32(r2, 23); + z2 = _mm256_xor_si256(z2, r2); + + y7 = z15; + y7 = _mm256_add_epi32(y7, z3); + r7 = y7; + y7 = _mm256_slli_epi32(y7, 9); + z7 = _mm256_xor_si256(z7, y7); + r7 = _mm256_srli_epi32(r7, 23); + z7 = _mm256_xor_si256(z7, r7); + + y6 = z14; + y6 = _mm256_add_epi32(y6, z2); + r6 = y6; + y6 = _mm256_slli_epi32(y6, 13); + z6 = _mm256_xor_si256(z6, y6); + r6 = _mm256_srli_epi32(r6, 19); + z6 = _mm256_xor_si256(z6, r6); + + y11 = z3; + y11 = _mm256_add_epi32(y11, z7); + r11 = y11; + y11 = _mm256_slli_epi32(y11, 13); + z11 = _mm256_xor_si256(z11, y11); + r11 = _mm256_srli_epi32(r11, 19); + z11 = _mm256_xor_si256(z11, r11); + + y10 = z2; + y10 = _mm256_add_epi32(y10, z6); + r10 = y10; + y10 = _mm256_slli_epi32(y10, 18); + z10 = _mm256_xor_si256(z10, y10); + r10 = _mm256_srli_epi32(r10, 14); + z10 = _mm256_xor_si256(z10, r10); + + y1 = z3; + y1 = _mm256_add_epi32(y1, z0); + r1 = y1; + y1 = _mm256_slli_epi32(y1, 7); + z1 = _mm256_xor_si256(z1, y1); + r1 = _mm256_srli_epi32(r1, 25); + z1 = _mm256_xor_si256(z1, r1); + + y15 = z7; + y15 = _mm256_add_epi32(y15, z11); + r15 = y15; + y15 = _mm256_slli_epi32(y15, 18); + z15 = _mm256_xor_si256(z15, y15); + r15 = _mm256_srli_epi32(r15, 14); + z15 = _mm256_xor_si256(z15, r15); + + y6 = z4; + y6 = _mm256_add_epi32(y6, z5); + r6 = y6; + y6 = _mm256_slli_epi32(y6, 7); + z6 = _mm256_xor_si256(z6, y6); + r6 = _mm256_srli_epi32(r6, 25); + z6 = _mm256_xor_si256(z6, r6); + + y2 = z0; + y2 = _mm256_add_epi32(y2, z1); + r2 = y2; + y2 = _mm256_slli_epi32(y2, 9); + z2 = _mm256_xor_si256(z2, y2); + r2 = _mm256_srli_epi32(r2, 23); + z2 = _mm256_xor_si256(z2, r2); + + y7 = z5; + y7 = _mm256_add_epi32(y7, z6); + r7 = y7; + y7 = _mm256_slli_epi32(y7, 9); + z7 = _mm256_xor_si256(z7, y7); + r7 = _mm256_srli_epi32(r7, 23); + z7 = _mm256_xor_si256(z7, r7); + + y3 = z1; + y3 = _mm256_add_epi32(y3, z2); + r3 = y3; + y3 = _mm256_slli_epi32(y3, 13); + z3 = _mm256_xor_si256(z3, y3); + r3 = _mm256_srli_epi32(r3, 19); + z3 = _mm256_xor_si256(z3, r3); + + y4 = z6; + y4 = _mm256_add_epi32(y4, z7); + r4 = y4; + y4 = _mm256_slli_epi32(y4, 13); + z4 = _mm256_xor_si256(z4, y4); + r4 = _mm256_srli_epi32(r4, 19); + z4 = _mm256_xor_si256(z4, r4); + + y0 = z2; + y0 = _mm256_add_epi32(y0, z3); + r0 = y0; + y0 = _mm256_slli_epi32(y0, 18); + z0 = _mm256_xor_si256(z0, y0); + r0 = _mm256_srli_epi32(r0, 14); + z0 = _mm256_xor_si256(z0, r0); + + y5 = z7; + y5 = _mm256_add_epi32(y5, z4); + r5 = y5; + y5 = _mm256_slli_epi32(y5, 18); + z5 = _mm256_xor_si256(z5, y5); + r5 = _mm256_srli_epi32(r5, 14); + z5 = _mm256_xor_si256(z5, r5); + + y11 = z9; + y11 = _mm256_add_epi32(y11, z10); + r11 = y11; + y11 = _mm256_slli_epi32(y11, 7); + z11 = _mm256_xor_si256(z11, y11); + r11 = _mm256_srli_epi32(r11, 25); + z11 = _mm256_xor_si256(z11, r11); + + y12 = z14; + y12 = _mm256_add_epi32(y12, z15); + r12 = y12; + y12 = _mm256_slli_epi32(y12, 7); + z12 = _mm256_xor_si256(z12, y12); + r12 = _mm256_srli_epi32(r12, 25); + z12 = _mm256_xor_si256(z12, r12); + + y8 = z10; + y8 = _mm256_add_epi32(y8, z11); + r8 = y8; + y8 = _mm256_slli_epi32(y8, 9); + z8 = _mm256_xor_si256(z8, y8); + r8 = _mm256_srli_epi32(r8, 23); + z8 = _mm256_xor_si256(z8, r8); + + y13 = z15; + y13 = _mm256_add_epi32(y13, z12); + r13 = y13; + y13 = _mm256_slli_epi32(y13, 9); + z13 = _mm256_xor_si256(z13, y13); + r13 = _mm256_srli_epi32(r13, 23); + z13 = _mm256_xor_si256(z13, r13); + + y9 = z11; + y9 = _mm256_add_epi32(y9, z8); + r9 = y9; + y9 = _mm256_slli_epi32(y9, 13); + z9 = _mm256_xor_si256(z9, y9); + r9 = _mm256_srli_epi32(r9, 19); + z9 = _mm256_xor_si256(z9, r9); + + y14 = z12; + y14 = _mm256_add_epi32(y14, z13); + r14 = y14; + y14 = _mm256_slli_epi32(y14, 13); + z14 = _mm256_xor_si256(z14, y14); + r14 = _mm256_srli_epi32(r14, 19); + z14 = _mm256_xor_si256(z14, r14); + + y10 = z8; + y10 = _mm256_add_epi32(y10, z9); + r10 = y10; + y10 = _mm256_slli_epi32(y10, 18); + z10 = _mm256_xor_si256(z10, y10); + r10 = _mm256_srli_epi32(r10, 14); + z10 = _mm256_xor_si256(z10, r10); + + y15 = z13; + y15 = _mm256_add_epi32(y15, z14); + r15 = y15; + y15 = _mm256_slli_epi32(y15, 18); + z15 = _mm256_xor_si256(z15, y15); + r15 = _mm256_srli_epi32(r15, 14); + z15 = _mm256_xor_si256(z15, r15); + } + +/* store data ; this macro first transpose data in-registers, and then store + * them in memory. much faster with icc. */ +#define ONEQUAD_TRANSPOSE(A, B, C, D) \ + { \ + __m128i t0, t1, t2, t3; \ + z##A = _mm256_add_epi32(z##A, orig##A); \ + z##B = _mm256_add_epi32(z##B, orig##B); \ + z##C = _mm256_add_epi32(z##C, orig##C); \ + z##D = _mm256_add_epi32(z##D, orig##D); \ + y##A = _mm256_unpacklo_epi32(z##A, z##B); \ + y##B = _mm256_unpacklo_epi32(z##C, z##D); \ + y##C = _mm256_unpackhi_epi32(z##A, z##B); \ + y##D = _mm256_unpackhi_epi32(z##C, z##D); \ + z##A = _mm256_unpacklo_epi64(y##A, y##B); \ + z##B = _mm256_unpackhi_epi64(y##A, y##B); \ + z##C = _mm256_unpacklo_epi64(y##C, y##D); \ + z##D = _mm256_unpackhi_epi64(y##C, y##D); \ + t0 = _mm_xor_si128(_mm256_extracti128_si256(z##A, 0), \ + _mm_loadu_si128((const __m128i*) (m + 0))); \ + _mm_storeu_si128((__m128i*) (c + 0), t0); \ + t1 = _mm_xor_si128(_mm256_extracti128_si256(z##B, 0), \ + _mm_loadu_si128((const __m128i*) (m + 64))); \ + _mm_storeu_si128((__m128i*) (c + 64), t1); \ + t2 = _mm_xor_si128(_mm256_extracti128_si256(z##C, 0), \ + _mm_loadu_si128((const __m128i*) (m + 128))); \ + _mm_storeu_si128((__m128i*) (c + 128), t2); \ + t3 = _mm_xor_si128(_mm256_extracti128_si256(z##D, 0), \ + _mm_loadu_si128((const __m128i*) (m + 192))); \ + _mm_storeu_si128((__m128i*) (c + 192), t3); \ + t0 = _mm_xor_si128(_mm256_extracti128_si256(z##A, 1), \ + _mm_loadu_si128((const __m128i*) (m + 256))); \ + _mm_storeu_si128((__m128i*) (c + 256), t0); \ + t1 = _mm_xor_si128(_mm256_extracti128_si256(z##B, 1), \ + _mm_loadu_si128((const __m128i*) (m + 320))); \ + _mm_storeu_si128((__m128i*) (c + 320), t1); \ + t2 = _mm_xor_si128(_mm256_extracti128_si256(z##C, 1), \ + _mm_loadu_si128((const __m128i*) (m + 384))); \ + _mm_storeu_si128((__m128i*) (c + 384), t2); \ + t3 = _mm_xor_si128(_mm256_extracti128_si256(z##D, 1), \ + _mm_loadu_si128((const __m128i*) (m + 448))); \ + _mm_storeu_si128((__m128i*) (c + 448), t3); \ + } + +#define ONEQUAD(A, B, C, D) ONEQUAD_TRANSPOSE(A, B, C, D) + +#define ONEQUAD_UNPCK(A, B, C, D) \ + { \ + z##A = _mm256_add_epi32(z##A, orig##A); \ + z##B = _mm256_add_epi32(z##B, orig##B); \ + z##C = _mm256_add_epi32(z##C, orig##C); \ + z##D = _mm256_add_epi32(z##D, orig##D); \ + y##A = _mm256_unpacklo_epi32(z##A, z##B); \ + y##B = _mm256_unpacklo_epi32(z##C, z##D); \ + y##C = _mm256_unpackhi_epi32(z##A, z##B); \ + y##D = _mm256_unpackhi_epi32(z##C, z##D); \ + z##A = _mm256_unpacklo_epi64(y##A, y##B); \ + z##B = _mm256_unpackhi_epi64(y##A, y##B); \ + z##C = _mm256_unpacklo_epi64(y##C, y##D); \ + z##D = _mm256_unpackhi_epi64(y##C, y##D); \ + } + +#define ONEOCTO(A, B, C, D, A2, B2, C2, D2) \ + { \ + ONEQUAD_UNPCK(A, B, C, D); \ + ONEQUAD_UNPCK(A2, B2, C2, D2); \ + y##A = _mm256_permute2x128_si256(z##A, z##A2, 0x20); \ + y##A2 = _mm256_permute2x128_si256(z##A, z##A2, 0x31); \ + y##B = _mm256_permute2x128_si256(z##B, z##B2, 0x20); \ + y##B2 = _mm256_permute2x128_si256(z##B, z##B2, 0x31); \ + y##C = _mm256_permute2x128_si256(z##C, z##C2, 0x20); \ + y##C2 = _mm256_permute2x128_si256(z##C, z##C2, 0x31); \ + y##D = _mm256_permute2x128_si256(z##D, z##D2, 0x20); \ + y##D2 = _mm256_permute2x128_si256(z##D, z##D2, 0x31); \ + y##A = _mm256_xor_si256(y##A, \ + _mm256_loadu_si256((const __m256i*) (m + 0))); \ + y##B = _mm256_xor_si256( \ + y##B, _mm256_loadu_si256((const __m256i*) (m + 64))); \ + y##C = _mm256_xor_si256( \ + y##C, _mm256_loadu_si256((const __m256i*) (m + 128))); \ + y##D = _mm256_xor_si256( \ + y##D, _mm256_loadu_si256((const __m256i*) (m + 192))); \ + y##A2 = _mm256_xor_si256( \ + y##A2, _mm256_loadu_si256((const __m256i*) (m + 256))); \ + y##B2 = _mm256_xor_si256( \ + y##B2, _mm256_loadu_si256((const __m256i*) (m + 320))); \ + y##C2 = _mm256_xor_si256( \ + y##C2, _mm256_loadu_si256((const __m256i*) (m + 384))); \ + y##D2 = _mm256_xor_si256( \ + y##D2, _mm256_loadu_si256((const __m256i*) (m + 448))); \ + _mm256_storeu_si256((__m256i*) (c + 0), y##A); \ + _mm256_storeu_si256((__m256i*) (c + 64), y##B); \ + _mm256_storeu_si256((__m256i*) (c + 128), y##C); \ + _mm256_storeu_si256((__m256i*) (c + 192), y##D); \ + _mm256_storeu_si256((__m256i*) (c + 256), y##A2); \ + _mm256_storeu_si256((__m256i*) (c + 320), y##B2); \ + _mm256_storeu_si256((__m256i*) (c + 384), y##C2); \ + _mm256_storeu_si256((__m256i*) (c + 448), y##D2); \ + } + + ONEOCTO(0, 1, 2, 3, 4, 5, 6, 7); + m += 32; + c += 32; + ONEOCTO(8, 9, 10, 11, 12, 13, 14, 15); + m -= 32; + c -= 32; + +#undef ONEQUAD +#undef ONEQUAD_TRANSPOSE +#undef ONEQUAD_UNPCK +#undef ONEOCTO + + bytes -= 512; + c += 512; + m += 512; + } +} diff --git a/windows/mbedtls/aes.h b/windows/mbedtls/aes.h deleted file mode 100644 index 00509808..00000000 --- a/windows/mbedtls/aes.h +++ /dev/null @@ -1,626 +0,0 @@ -/** - * \file aes.h - * - * \brief This file contains AES definitions and functions. - * - * The Advanced Encryption Standard (AES) specifies a FIPS-approved - * cryptographic algorithm that can be used to protect electronic - * data. - * - * The AES algorithm is a symmetric block cipher that can - * encrypt and decrypt information. For more information, see - * FIPS Publication 197: Advanced Encryption Standard and - * ISO/IEC 18033-2:2006: Information technology -- Security - * techniques -- Encryption algorithms -- Part 2: Asymmetric - * ciphers. - * - * The AES-XTS block mode is standardized by NIST SP 800-38E - * - * and described in detail by IEEE P1619 - * . - */ - -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_AES_H -#define MBEDTLS_AES_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -/* padlock.c and aesni.c rely on these values! */ -#define MBEDTLS_AES_ENCRYPT 1 /**< AES encryption. */ -#define MBEDTLS_AES_DECRYPT 0 /**< AES decryption. */ - -/* Error codes in range 0x0020-0x0022 */ -#define MBEDTLS_ERR_AES_INVALID_KEY_LENGTH -0x0020 /**< Invalid key length. */ -#define MBEDTLS_ERR_AES_INVALID_INPUT_LENGTH -0x0022 /**< Invalid data input length. */ - -/* Error codes in range 0x0021-0x0025 */ -#define MBEDTLS_ERR_AES_BAD_INPUT_DATA -0x0021 /**< Invalid input data. */ - -#if ( defined(__ARMCC_VERSION) || defined(_MSC_VER) ) && \ - !defined(inline) && !defined(__cplusplus) -#define inline __inline -#endif - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_AES_ALT) -// Regular implementation -// - -/** - * \brief The AES context-type definition. - */ -typedef struct mbedtls_aes_context -{ - int MBEDTLS_PRIVATE(nr); /*!< The number of rounds. */ - uint32_t *MBEDTLS_PRIVATE(rk); /*!< AES round keys. */ - uint32_t MBEDTLS_PRIVATE(buf)[68]; /*!< Unaligned data buffer. This buffer can - hold 32 extra Bytes, which can be used for - one of the following purposes: -
  • Alignment if VIA padlock is - used.
  • -
  • Simplifying key expansion in the 256-bit - case by generating an extra round key. -
*/ -} -mbedtls_aes_context; - -#if defined(MBEDTLS_CIPHER_MODE_XTS) -/** - * \brief The AES XTS context-type definition. - */ -typedef struct mbedtls_aes_xts_context -{ - mbedtls_aes_context MBEDTLS_PRIVATE(crypt); /*!< The AES context to use for AES block - encryption or decryption. */ - mbedtls_aes_context MBEDTLS_PRIVATE(tweak); /*!< The AES context used for tweak - computation. */ -} mbedtls_aes_xts_context; -#endif /* MBEDTLS_CIPHER_MODE_XTS */ - -#else /* MBEDTLS_AES_ALT */ -#include "aes_alt.h" -#endif /* MBEDTLS_AES_ALT */ - -/** - * \brief This function initializes the specified AES context. - * - * It must be the first API called before using - * the context. - * - * \param ctx The AES context to initialize. This must not be \c NULL. - */ -void mbedtls_aes_init( mbedtls_aes_context *ctx ); - -/** - * \brief This function releases and clears the specified AES context. - * - * \param ctx The AES context to clear. - * If this is \c NULL, this function does nothing. - * Otherwise, the context must have been at least initialized. - */ -void mbedtls_aes_free( mbedtls_aes_context *ctx ); - -#if defined(MBEDTLS_CIPHER_MODE_XTS) -/** - * \brief This function initializes the specified AES XTS context. - * - * It must be the first API called before using - * the context. - * - * \param ctx The AES XTS context to initialize. This must not be \c NULL. - */ -void mbedtls_aes_xts_init( mbedtls_aes_xts_context *ctx ); - -/** - * \brief This function releases and clears the specified AES XTS context. - * - * \param ctx The AES XTS context to clear. - * If this is \c NULL, this function does nothing. - * Otherwise, the context must have been at least initialized. - */ -void mbedtls_aes_xts_free( mbedtls_aes_xts_context *ctx ); -#endif /* MBEDTLS_CIPHER_MODE_XTS */ - -/** - * \brief This function sets the encryption key. - * - * \param ctx The AES context to which the key should be bound. - * It must be initialized. - * \param key The encryption key. - * This must be a readable buffer of size \p keybits bits. - * \param keybits The size of data passed in bits. Valid options are: - *
  • 128 bits
  • - *
  • 192 bits
  • - *
  • 256 bits
- * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_AES_INVALID_KEY_LENGTH on failure. - */ -int mbedtls_aes_setkey_enc( mbedtls_aes_context *ctx, const unsigned char *key, - unsigned int keybits ); - -/** - * \brief This function sets the decryption key. - * - * \param ctx The AES context to which the key should be bound. - * It must be initialized. - * \param key The decryption key. - * This must be a readable buffer of size \p keybits bits. - * \param keybits The size of data passed. Valid options are: - *
  • 128 bits
  • - *
  • 192 bits
  • - *
  • 256 bits
- * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_AES_INVALID_KEY_LENGTH on failure. - */ -int mbedtls_aes_setkey_dec( mbedtls_aes_context *ctx, const unsigned char *key, - unsigned int keybits ); - -#if defined(MBEDTLS_CIPHER_MODE_XTS) -/** - * \brief This function prepares an XTS context for encryption and - * sets the encryption key. - * - * \param ctx The AES XTS context to which the key should be bound. - * It must be initialized. - * \param key The encryption key. This is comprised of the XTS key1 - * concatenated with the XTS key2. - * This must be a readable buffer of size \p keybits bits. - * \param keybits The size of \p key passed in bits. Valid options are: - *
  • 256 bits (each of key1 and key2 is a 128-bit key)
  • - *
  • 512 bits (each of key1 and key2 is a 256-bit key)
- * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_AES_INVALID_KEY_LENGTH on failure. - */ -int mbedtls_aes_xts_setkey_enc( mbedtls_aes_xts_context *ctx, - const unsigned char *key, - unsigned int keybits ); - -/** - * \brief This function prepares an XTS context for decryption and - * sets the decryption key. - * - * \param ctx The AES XTS context to which the key should be bound. - * It must be initialized. - * \param key The decryption key. This is comprised of the XTS key1 - * concatenated with the XTS key2. - * This must be a readable buffer of size \p keybits bits. - * \param keybits The size of \p key passed in bits. Valid options are: - *
  • 256 bits (each of key1 and key2 is a 128-bit key)
  • - *
  • 512 bits (each of key1 and key2 is a 256-bit key)
- * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_AES_INVALID_KEY_LENGTH on failure. - */ -int mbedtls_aes_xts_setkey_dec( mbedtls_aes_xts_context *ctx, - const unsigned char *key, - unsigned int keybits ); -#endif /* MBEDTLS_CIPHER_MODE_XTS */ - -/** - * \brief This function performs an AES single-block encryption or - * decryption operation. - * - * It performs the operation defined in the \p mode parameter - * (encrypt or decrypt), on the input data buffer defined in - * the \p input parameter. - * - * mbedtls_aes_init(), and either mbedtls_aes_setkey_enc() or - * mbedtls_aes_setkey_dec() must be called before the first - * call to this API with the same context. - * - * \param ctx The AES context to use for encryption or decryption. - * It must be initialized and bound to a key. - * \param mode The AES operation: #MBEDTLS_AES_ENCRYPT or - * #MBEDTLS_AES_DECRYPT. - * \param input The buffer holding the input data. - * It must be readable and at least \c 16 Bytes long. - * \param output The buffer where the output data will be written. - * It must be writeable and at least \c 16 Bytes long. - - * \return \c 0 on success. - */ -int mbedtls_aes_crypt_ecb( mbedtls_aes_context *ctx, - int mode, - const unsigned char input[16], - unsigned char output[16] ); - -#if defined(MBEDTLS_CIPHER_MODE_CBC) -/** - * \brief This function performs an AES-CBC encryption or decryption operation - * on full blocks. - * - * It performs the operation defined in the \p mode - * parameter (encrypt/decrypt), on the input data buffer defined in - * the \p input parameter. - * - * It can be called as many times as needed, until all the input - * data is processed. mbedtls_aes_init(), and either - * mbedtls_aes_setkey_enc() or mbedtls_aes_setkey_dec() must be called - * before the first call to this API with the same context. - * - * \note This function operates on full blocks, that is, the input size - * must be a multiple of the AES block size of \c 16 Bytes. - * - * \note Upon exit, the content of the IV is updated so that you can - * call the same function again on the next - * block(s) of data and get the same result as if it was - * encrypted in one call. This allows a "streaming" usage. - * If you need to retain the contents of the IV, you should - * either save it manually or use the cipher module instead. - * - * - * \param ctx The AES context to use for encryption or decryption. - * It must be initialized and bound to a key. - * \param mode The AES operation: #MBEDTLS_AES_ENCRYPT or - * #MBEDTLS_AES_DECRYPT. - * \param length The length of the input data in Bytes. This must be a - * multiple of the block size (\c 16 Bytes). - * \param iv Initialization vector (updated after use). - * It must be a readable and writeable buffer of \c 16 Bytes. - * \param input The buffer holding the input data. - * It must be readable and of size \p length Bytes. - * \param output The buffer holding the output data. - * It must be writeable and of size \p length Bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_AES_INVALID_INPUT_LENGTH - * on failure. - */ -int mbedtls_aes_crypt_cbc( mbedtls_aes_context *ctx, - int mode, - size_t length, - unsigned char iv[16], - const unsigned char *input, - unsigned char *output ); -#endif /* MBEDTLS_CIPHER_MODE_CBC */ - -#if defined(MBEDTLS_CIPHER_MODE_XTS) -/** - * \brief This function performs an AES-XTS encryption or decryption - * operation for an entire XTS data unit. - * - * AES-XTS encrypts or decrypts blocks based on their location as - * defined by a data unit number. The data unit number must be - * provided by \p data_unit. - * - * NIST SP 800-38E limits the maximum size of a data unit to 2^20 - * AES blocks. If the data unit is larger than this, this function - * returns #MBEDTLS_ERR_AES_INVALID_INPUT_LENGTH. - * - * \param ctx The AES XTS context to use for AES XTS operations. - * It must be initialized and bound to a key. - * \param mode The AES operation: #MBEDTLS_AES_ENCRYPT or - * #MBEDTLS_AES_DECRYPT. - * \param length The length of a data unit in Bytes. This can be any - * length between 16 bytes and 2^24 bytes inclusive - * (between 1 and 2^20 block cipher blocks). - * \param data_unit The address of the data unit encoded as an array of 16 - * bytes in little-endian format. For disk encryption, this - * is typically the index of the block device sector that - * contains the data. - * \param input The buffer holding the input data (which is an entire - * data unit). This function reads \p length Bytes from \p - * input. - * \param output The buffer holding the output data (which is an entire - * data unit). This function writes \p length Bytes to \p - * output. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_AES_INVALID_INPUT_LENGTH if \p length is - * smaller than an AES block in size (16 Bytes) or if \p - * length is larger than 2^20 blocks (16 MiB). - */ -int mbedtls_aes_crypt_xts( mbedtls_aes_xts_context *ctx, - int mode, - size_t length, - const unsigned char data_unit[16], - const unsigned char *input, - unsigned char *output ); -#endif /* MBEDTLS_CIPHER_MODE_XTS */ - -#if defined(MBEDTLS_CIPHER_MODE_CFB) -/** - * \brief This function performs an AES-CFB128 encryption or decryption - * operation. - * - * It performs the operation defined in the \p mode - * parameter (encrypt or decrypt), on the input data buffer - * defined in the \p input parameter. - * - * For CFB, you must set up the context with mbedtls_aes_setkey_enc(), - * regardless of whether you are performing an encryption or decryption - * operation, that is, regardless of the \p mode parameter. This is - * because CFB mode uses the same key schedule for encryption and - * decryption. - * - * \note Upon exit, the content of the IV is updated so that you can - * call the same function again on the next - * block(s) of data and get the same result as if it was - * encrypted in one call. This allows a "streaming" usage. - * If you need to retain the contents of the - * IV, you must either save it manually or use the cipher - * module instead. - * - * - * \param ctx The AES context to use for encryption or decryption. - * It must be initialized and bound to a key. - * \param mode The AES operation: #MBEDTLS_AES_ENCRYPT or - * #MBEDTLS_AES_DECRYPT. - * \param length The length of the input data in Bytes. - * \param iv_off The offset in IV (updated after use). - * It must point to a valid \c size_t. - * \param iv The initialization vector (updated after use). - * It must be a readable and writeable buffer of \c 16 Bytes. - * \param input The buffer holding the input data. - * It must be readable and of size \p length Bytes. - * \param output The buffer holding the output data. - * It must be writeable and of size \p length Bytes. - * - * \return \c 0 on success. - */ -int mbedtls_aes_crypt_cfb128( mbedtls_aes_context *ctx, - int mode, - size_t length, - size_t *iv_off, - unsigned char iv[16], - const unsigned char *input, - unsigned char *output ); - -/** - * \brief This function performs an AES-CFB8 encryption or decryption - * operation. - * - * It performs the operation defined in the \p mode - * parameter (encrypt/decrypt), on the input data buffer defined - * in the \p input parameter. - * - * Due to the nature of CFB, you must use the same key schedule for - * both encryption and decryption operations. Therefore, you must - * use the context initialized with mbedtls_aes_setkey_enc() for - * both #MBEDTLS_AES_ENCRYPT and #MBEDTLS_AES_DECRYPT. - * - * \note Upon exit, the content of the IV is updated so that you can - * call the same function again on the next - * block(s) of data and get the same result as if it was - * encrypted in one call. This allows a "streaming" usage. - * If you need to retain the contents of the - * IV, you should either save it manually or use the cipher - * module instead. - * - * - * \param ctx The AES context to use for encryption or decryption. - * It must be initialized and bound to a key. - * \param mode The AES operation: #MBEDTLS_AES_ENCRYPT or - * #MBEDTLS_AES_DECRYPT - * \param length The length of the input data. - * \param iv The initialization vector (updated after use). - * It must be a readable and writeable buffer of \c 16 Bytes. - * \param input The buffer holding the input data. - * It must be readable and of size \p length Bytes. - * \param output The buffer holding the output data. - * It must be writeable and of size \p length Bytes. - * - * \return \c 0 on success. - */ -int mbedtls_aes_crypt_cfb8( mbedtls_aes_context *ctx, - int mode, - size_t length, - unsigned char iv[16], - const unsigned char *input, - unsigned char *output ); -#endif /*MBEDTLS_CIPHER_MODE_CFB */ - -#if defined(MBEDTLS_CIPHER_MODE_OFB) -/** - * \brief This function performs an AES-OFB (Output Feedback Mode) - * encryption or decryption operation. - * - * For OFB, you must set up the context with - * mbedtls_aes_setkey_enc(), regardless of whether you are - * performing an encryption or decryption operation. This is - * because OFB mode uses the same key schedule for encryption and - * decryption. - * - * The OFB operation is identical for encryption or decryption, - * therefore no operation mode needs to be specified. - * - * \note Upon exit, the content of iv, the Initialisation Vector, is - * updated so that you can call the same function again on the next - * block(s) of data and get the same result as if it was encrypted - * in one call. This allows a "streaming" usage, by initialising - * iv_off to 0 before the first call, and preserving its value - * between calls. - * - * For non-streaming use, the iv should be initialised on each call - * to a unique value, and iv_off set to 0 on each call. - * - * If you need to retain the contents of the initialisation vector, - * you must either save it manually or use the cipher module - * instead. - * - * \warning For the OFB mode, the initialisation vector must be unique - * every encryption operation. Reuse of an initialisation vector - * will compromise security. - * - * \param ctx The AES context to use for encryption or decryption. - * It must be initialized and bound to a key. - * \param length The length of the input data. - * \param iv_off The offset in IV (updated after use). - * It must point to a valid \c size_t. - * \param iv The initialization vector (updated after use). - * It must be a readable and writeable buffer of \c 16 Bytes. - * \param input The buffer holding the input data. - * It must be readable and of size \p length Bytes. - * \param output The buffer holding the output data. - * It must be writeable and of size \p length Bytes. - * - * \return \c 0 on success. - */ -int mbedtls_aes_crypt_ofb( mbedtls_aes_context *ctx, - size_t length, - size_t *iv_off, - unsigned char iv[16], - const unsigned char *input, - unsigned char *output ); - -#endif /* MBEDTLS_CIPHER_MODE_OFB */ - -#if defined(MBEDTLS_CIPHER_MODE_CTR) -/** - * \brief This function performs an AES-CTR encryption or decryption - * operation. - * - * This function performs the operation defined in the \p mode - * parameter (encrypt/decrypt), on the input data buffer - * defined in the \p input parameter. - * - * Due to the nature of CTR, you must use the same key schedule - * for both encryption and decryption operations. Therefore, you - * must use the context initialized with mbedtls_aes_setkey_enc() - * for both #MBEDTLS_AES_ENCRYPT and #MBEDTLS_AES_DECRYPT. - * - * \warning You must never reuse a nonce value with the same key. Doing so - * would void the encryption for the two messages encrypted with - * the same nonce and key. - * - * There are two common strategies for managing nonces with CTR: - * - * 1. You can handle everything as a single message processed over - * successive calls to this function. In that case, you want to - * set \p nonce_counter and \p nc_off to 0 for the first call, and - * then preserve the values of \p nonce_counter, \p nc_off and \p - * stream_block across calls to this function as they will be - * updated by this function. - * - * With this strategy, you must not encrypt more than 2**128 - * blocks of data with the same key. - * - * 2. You can encrypt separate messages by dividing the \p - * nonce_counter buffer in two areas: the first one used for a - * per-message nonce, handled by yourself, and the second one - * updated by this function internally. - * - * For example, you might reserve the first 12 bytes for the - * per-message nonce, and the last 4 bytes for internal use. In that - * case, before calling this function on a new message you need to - * set the first 12 bytes of \p nonce_counter to your chosen nonce - * value, the last 4 to 0, and \p nc_off to 0 (which will cause \p - * stream_block to be ignored). That way, you can encrypt at most - * 2**96 messages of up to 2**32 blocks each with the same key. - * - * The per-message nonce (or information sufficient to reconstruct - * it) needs to be communicated with the ciphertext and must be unique. - * The recommended way to ensure uniqueness is to use a message - * counter. An alternative is to generate random nonces, but this - * limits the number of messages that can be securely encrypted: - * for example, with 96-bit random nonces, you should not encrypt - * more than 2**32 messages with the same key. - * - * Note that for both stategies, sizes are measured in blocks and - * that an AES block is 16 bytes. - * - * \warning Upon return, \p stream_block contains sensitive data. Its - * content must not be written to insecure storage and should be - * securely discarded as soon as it's no longer needed. - * - * \param ctx The AES context to use for encryption or decryption. - * It must be initialized and bound to a key. - * \param length The length of the input data. - * \param nc_off The offset in the current \p stream_block, for - * resuming within the current cipher stream. The - * offset pointer should be 0 at the start of a stream. - * It must point to a valid \c size_t. - * \param nonce_counter The 128-bit nonce and counter. - * It must be a readable-writeable buffer of \c 16 Bytes. - * \param stream_block The saved stream block for resuming. This is - * overwritten by the function. - * It must be a readable-writeable buffer of \c 16 Bytes. - * \param input The buffer holding the input data. - * It must be readable and of size \p length Bytes. - * \param output The buffer holding the output data. - * It must be writeable and of size \p length Bytes. - * - * \return \c 0 on success. - */ -int mbedtls_aes_crypt_ctr( mbedtls_aes_context *ctx, - size_t length, - size_t *nc_off, - unsigned char nonce_counter[16], - unsigned char stream_block[16], - const unsigned char *input, - unsigned char *output ); -#endif /* MBEDTLS_CIPHER_MODE_CTR */ - -/** - * \brief Internal AES block encryption function. This is only - * exposed to allow overriding it using - * \c MBEDTLS_AES_ENCRYPT_ALT. - * - * \param ctx The AES context to use for encryption. - * \param input The plaintext block. - * \param output The output (ciphertext) block. - * - * \return \c 0 on success. - */ -int mbedtls_internal_aes_encrypt( mbedtls_aes_context *ctx, - const unsigned char input[16], - unsigned char output[16] ); - -/** - * \brief Internal AES block decryption function. This is only - * exposed to allow overriding it using see - * \c MBEDTLS_AES_DECRYPT_ALT. - * - * \param ctx The AES context to use for decryption. - * \param input The ciphertext block. - * \param output The output (plaintext) block. - * - * \return \c 0 on success. - */ -int mbedtls_internal_aes_decrypt( mbedtls_aes_context *ctx, - const unsigned char input[16], - unsigned char output[16] ); - -#if defined(MBEDTLS_SELF_TEST) -/** - * \brief Checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_aes_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* aes.h */ diff --git a/windows/mbedtls/aria.h b/windows/mbedtls/aria.h deleted file mode 100644 index 08547068..00000000 --- a/windows/mbedtls/aria.h +++ /dev/null @@ -1,356 +0,0 @@ -/** - * \file aria.h - * - * \brief ARIA block cipher - * - * The ARIA algorithm is a symmetric block cipher that can encrypt and - * decrypt information. It is defined by the Korean Agency for - * Technology and Standards (KATS) in KS X 1213:2004 (in - * Korean, but see http://210.104.33.10/ARIA/index-e.html in English) - * and also described by the IETF in RFC 5794. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_ARIA_H -#define MBEDTLS_ARIA_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -#include "mbedtls/platform_util.h" - -#define MBEDTLS_ARIA_ENCRYPT 1 /**< ARIA encryption. */ -#define MBEDTLS_ARIA_DECRYPT 0 /**< ARIA decryption. */ - -#define MBEDTLS_ARIA_BLOCKSIZE 16 /**< ARIA block size in bytes. */ -#define MBEDTLS_ARIA_MAX_ROUNDS 16 /**< Maxiumum number of rounds in ARIA. */ -#define MBEDTLS_ARIA_MAX_KEYSIZE 32 /**< Maximum size of an ARIA key in bytes. */ - -#define MBEDTLS_ERR_ARIA_BAD_INPUT_DATA -0x005C /**< Bad input data. */ - -#define MBEDTLS_ERR_ARIA_INVALID_INPUT_LENGTH -0x005E /**< Invalid data input length. */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_ARIA_ALT) -// Regular implementation -// - -/** - * \brief The ARIA context-type definition. - */ -typedef struct mbedtls_aria_context -{ - unsigned char MBEDTLS_PRIVATE(nr); /*!< The number of rounds (12, 14 or 16) */ - /*! The ARIA round keys. */ - uint32_t MBEDTLS_PRIVATE(rk)[MBEDTLS_ARIA_MAX_ROUNDS + 1][MBEDTLS_ARIA_BLOCKSIZE / 4]; -} -mbedtls_aria_context; - -#else /* MBEDTLS_ARIA_ALT */ -#include "aria_alt.h" -#endif /* MBEDTLS_ARIA_ALT */ - -/** - * \brief This function initializes the specified ARIA context. - * - * It must be the first API called before using - * the context. - * - * \param ctx The ARIA context to initialize. This must not be \c NULL. - */ -void mbedtls_aria_init( mbedtls_aria_context *ctx ); - -/** - * \brief This function releases and clears the specified ARIA context. - * - * \param ctx The ARIA context to clear. This may be \c NULL, in which - * case this function returns immediately. If it is not \c NULL, - * it must point to an initialized ARIA context. - */ -void mbedtls_aria_free( mbedtls_aria_context *ctx ); - -/** - * \brief This function sets the encryption key. - * - * \param ctx The ARIA context to which the key should be bound. - * This must be initialized. - * \param key The encryption key. This must be a readable buffer - * of size \p keybits Bits. - * \param keybits The size of \p key in Bits. Valid options are: - *
  • 128 bits
  • - *
  • 192 bits
  • - *
  • 256 bits
- * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_aria_setkey_enc( mbedtls_aria_context *ctx, - const unsigned char *key, - unsigned int keybits ); - -/** - * \brief This function sets the decryption key. - * - * \param ctx The ARIA context to which the key should be bound. - * This must be initialized. - * \param key The decryption key. This must be a readable buffer - * of size \p keybits Bits. - * \param keybits The size of data passed. Valid options are: - *
  • 128 bits
  • - *
  • 192 bits
  • - *
  • 256 bits
- * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_aria_setkey_dec( mbedtls_aria_context *ctx, - const unsigned char *key, - unsigned int keybits ); - -/** - * \brief This function performs an ARIA single-block encryption or - * decryption operation. - * - * It performs encryption or decryption (depending on whether - * the key was set for encryption on decryption) on the input - * data buffer defined in the \p input parameter. - * - * mbedtls_aria_init(), and either mbedtls_aria_setkey_enc() or - * mbedtls_aria_setkey_dec() must be called before the first - * call to this API with the same context. - * - * \param ctx The ARIA context to use for encryption or decryption. - * This must be initialized and bound to a key. - * \param input The 16-Byte buffer holding the input data. - * \param output The 16-Byte buffer holding the output data. - - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_aria_crypt_ecb( mbedtls_aria_context *ctx, - const unsigned char input[MBEDTLS_ARIA_BLOCKSIZE], - unsigned char output[MBEDTLS_ARIA_BLOCKSIZE] ); - -#if defined(MBEDTLS_CIPHER_MODE_CBC) -/** - * \brief This function performs an ARIA-CBC encryption or decryption operation - * on full blocks. - * - * It performs the operation defined in the \p mode - * parameter (encrypt/decrypt), on the input data buffer defined in - * the \p input parameter. - * - * It can be called as many times as needed, until all the input - * data is processed. mbedtls_aria_init(), and either - * mbedtls_aria_setkey_enc() or mbedtls_aria_setkey_dec() must be called - * before the first call to this API with the same context. - * - * \note This function operates on aligned blocks, that is, the input size - * must be a multiple of the ARIA block size of 16 Bytes. - * - * \note Upon exit, the content of the IV is updated so that you can - * call the same function again on the next - * block(s) of data and get the same result as if it was - * encrypted in one call. This allows a "streaming" usage. - * If you need to retain the contents of the IV, you should - * either save it manually or use the cipher module instead. - * - * - * \param ctx The ARIA context to use for encryption or decryption. - * This must be initialized and bound to a key. - * \param mode The mode of operation. This must be either - * #MBEDTLS_ARIA_ENCRYPT for encryption, or - * #MBEDTLS_ARIA_DECRYPT for decryption. - * \param length The length of the input data in Bytes. This must be a - * multiple of the block size (16 Bytes). - * \param iv Initialization vector (updated after use). - * This must be a readable buffer of size 16 Bytes. - * \param input The buffer holding the input data. This must - * be a readable buffer of length \p length Bytes. - * \param output The buffer holding the output data. This must - * be a writable buffer of length \p length Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_aria_crypt_cbc( mbedtls_aria_context *ctx, - int mode, - size_t length, - unsigned char iv[MBEDTLS_ARIA_BLOCKSIZE], - const unsigned char *input, - unsigned char *output ); -#endif /* MBEDTLS_CIPHER_MODE_CBC */ - -#if defined(MBEDTLS_CIPHER_MODE_CFB) -/** - * \brief This function performs an ARIA-CFB128 encryption or decryption - * operation. - * - * It performs the operation defined in the \p mode - * parameter (encrypt or decrypt), on the input data buffer - * defined in the \p input parameter. - * - * For CFB, you must set up the context with mbedtls_aria_setkey_enc(), - * regardless of whether you are performing an encryption or decryption - * operation, that is, regardless of the \p mode parameter. This is - * because CFB mode uses the same key schedule for encryption and - * decryption. - * - * \note Upon exit, the content of the IV is updated so that you can - * call the same function again on the next - * block(s) of data and get the same result as if it was - * encrypted in one call. This allows a "streaming" usage. - * If you need to retain the contents of the - * IV, you must either save it manually or use the cipher - * module instead. - * - * - * \param ctx The ARIA context to use for encryption or decryption. - * This must be initialized and bound to a key. - * \param mode The mode of operation. This must be either - * #MBEDTLS_ARIA_ENCRYPT for encryption, or - * #MBEDTLS_ARIA_DECRYPT for decryption. - * \param length The length of the input data \p input in Bytes. - * \param iv_off The offset in IV (updated after use). - * This must not be larger than 15. - * \param iv The initialization vector (updated after use). - * This must be a readable buffer of size 16 Bytes. - * \param input The buffer holding the input data. This must - * be a readable buffer of length \p length Bytes. - * \param output The buffer holding the output data. This must - * be a writable buffer of length \p length Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_aria_crypt_cfb128( mbedtls_aria_context *ctx, - int mode, - size_t length, - size_t *iv_off, - unsigned char iv[MBEDTLS_ARIA_BLOCKSIZE], - const unsigned char *input, - unsigned char *output ); -#endif /* MBEDTLS_CIPHER_MODE_CFB */ - -#if defined(MBEDTLS_CIPHER_MODE_CTR) -/** - * \brief This function performs an ARIA-CTR encryption or decryption - * operation. - * - * This function performs the operation defined in the \p mode - * parameter (encrypt/decrypt), on the input data buffer - * defined in the \p input parameter. - * - * Due to the nature of CTR, you must use the same key schedule - * for both encryption and decryption operations. Therefore, you - * must use the context initialized with mbedtls_aria_setkey_enc() - * for both #MBEDTLS_ARIA_ENCRYPT and #MBEDTLS_ARIA_DECRYPT. - * - * \warning You must never reuse a nonce value with the same key. Doing so - * would void the encryption for the two messages encrypted with - * the same nonce and key. - * - * There are two common strategies for managing nonces with CTR: - * - * 1. You can handle everything as a single message processed over - * successive calls to this function. In that case, you want to - * set \p nonce_counter and \p nc_off to 0 for the first call, and - * then preserve the values of \p nonce_counter, \p nc_off and \p - * stream_block across calls to this function as they will be - * updated by this function. - * - * With this strategy, you must not encrypt more than 2**128 - * blocks of data with the same key. - * - * 2. You can encrypt separate messages by dividing the \p - * nonce_counter buffer in two areas: the first one used for a - * per-message nonce, handled by yourself, and the second one - * updated by this function internally. - * - * For example, you might reserve the first 12 bytes for the - * per-message nonce, and the last 4 bytes for internal use. In that - * case, before calling this function on a new message you need to - * set the first 12 bytes of \p nonce_counter to your chosen nonce - * value, the last 4 to 0, and \p nc_off to 0 (which will cause \p - * stream_block to be ignored). That way, you can encrypt at most - * 2**96 messages of up to 2**32 blocks each with the same key. - * - * The per-message nonce (or information sufficient to reconstruct - * it) needs to be communicated with the ciphertext and must be unique. - * The recommended way to ensure uniqueness is to use a message - * counter. An alternative is to generate random nonces, but this - * limits the number of messages that can be securely encrypted: - * for example, with 96-bit random nonces, you should not encrypt - * more than 2**32 messages with the same key. - * - * Note that for both stategies, sizes are measured in blocks and - * that an ARIA block is 16 bytes. - * - * \warning Upon return, \p stream_block contains sensitive data. Its - * content must not be written to insecure storage and should be - * securely discarded as soon as it's no longer needed. - * - * \param ctx The ARIA context to use for encryption or decryption. - * This must be initialized and bound to a key. - * \param length The length of the input data \p input in Bytes. - * \param nc_off The offset in Bytes in the current \p stream_block, - * for resuming within the current cipher stream. The - * offset pointer should be \c 0 at the start of a - * stream. This must not be larger than \c 15 Bytes. - * \param nonce_counter The 128-bit nonce and counter. This must point to - * a read/write buffer of length \c 16 bytes. - * \param stream_block The saved stream block for resuming. This must - * point to a read/write buffer of length \c 16 bytes. - * This is overwritten by the function. - * \param input The buffer holding the input data. This must - * be a readable buffer of length \p length Bytes. - * \param output The buffer holding the output data. This must - * be a writable buffer of length \p length Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_aria_crypt_ctr( mbedtls_aria_context *ctx, - size_t length, - size_t *nc_off, - unsigned char nonce_counter[MBEDTLS_ARIA_BLOCKSIZE], - unsigned char stream_block[MBEDTLS_ARIA_BLOCKSIZE], - const unsigned char *input, - unsigned char *output ); -#endif /* MBEDTLS_CIPHER_MODE_CTR */ - -#if defined(MBEDTLS_SELF_TEST) -/** - * \brief Checkup routine. - * - * \return \c 0 on success, or \c 1 on failure. - */ -int mbedtls_aria_self_test( int verbose ); -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* aria.h */ diff --git a/windows/mbedtls/asn1.h b/windows/mbedtls/asn1.h deleted file mode 100644 index 46685810..00000000 --- a/windows/mbedtls/asn1.h +++ /dev/null @@ -1,604 +0,0 @@ -/** - * \file asn1.h - * - * \brief Generic ASN.1 parsing - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_ASN1_H -#define MBEDTLS_ASN1_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include - -#if defined(MBEDTLS_BIGNUM_C) -#include "mbedtls/bignum.h" -#endif - -/** - * \addtogroup asn1_module - * \{ - */ - -/** - * \name ASN1 Error codes - * These error codes are OR'ed to X509 error codes for - * higher error granularity. - * ASN1 is a standard to specify data structures. - * \{ - */ -#define MBEDTLS_ERR_ASN1_OUT_OF_DATA -0x0060 /**< Out of data when parsing an ASN1 data structure. */ -#define MBEDTLS_ERR_ASN1_UNEXPECTED_TAG -0x0062 /**< ASN1 tag was of an unexpected value. */ -#define MBEDTLS_ERR_ASN1_INVALID_LENGTH -0x0064 /**< Error when trying to determine the length or invalid length. */ -#define MBEDTLS_ERR_ASN1_LENGTH_MISMATCH -0x0066 /**< Actual length differs from expected length. */ -#define MBEDTLS_ERR_ASN1_INVALID_DATA -0x0068 /**< Data is invalid. */ -#define MBEDTLS_ERR_ASN1_ALLOC_FAILED -0x006A /**< Memory allocation failed */ -#define MBEDTLS_ERR_ASN1_BUF_TOO_SMALL -0x006C /**< Buffer too small when writing ASN.1 data structure. */ - -/* \} name */ - -/** - * \name DER constants - * These constants comply with the DER encoded ASN.1 type tags. - * DER encoding uses hexadecimal representation. - * An example DER sequence is:\n - * - 0x02 -- tag indicating INTEGER - * - 0x01 -- length in octets - * - 0x05 -- value - * Such sequences are typically read into \c ::mbedtls_x509_buf. - * \{ - */ -#define MBEDTLS_ASN1_BOOLEAN 0x01 -#define MBEDTLS_ASN1_INTEGER 0x02 -#define MBEDTLS_ASN1_BIT_STRING 0x03 -#define MBEDTLS_ASN1_OCTET_STRING 0x04 -#define MBEDTLS_ASN1_NULL 0x05 -#define MBEDTLS_ASN1_OID 0x06 -#define MBEDTLS_ASN1_ENUMERATED 0x0A -#define MBEDTLS_ASN1_UTF8_STRING 0x0C -#define MBEDTLS_ASN1_SEQUENCE 0x10 -#define MBEDTLS_ASN1_SET 0x11 -#define MBEDTLS_ASN1_PRINTABLE_STRING 0x13 -#define MBEDTLS_ASN1_T61_STRING 0x14 -#define MBEDTLS_ASN1_IA5_STRING 0x16 -#define MBEDTLS_ASN1_UTC_TIME 0x17 -#define MBEDTLS_ASN1_GENERALIZED_TIME 0x18 -#define MBEDTLS_ASN1_UNIVERSAL_STRING 0x1C -#define MBEDTLS_ASN1_BMP_STRING 0x1E -#define MBEDTLS_ASN1_PRIMITIVE 0x00 -#define MBEDTLS_ASN1_CONSTRUCTED 0x20 -#define MBEDTLS_ASN1_CONTEXT_SPECIFIC 0x80 - -/* Slightly smaller way to check if tag is a string tag - * compared to canonical implementation. */ -#define MBEDTLS_ASN1_IS_STRING_TAG( tag ) \ - ( ( tag ) < 32u && ( \ - ( ( 1u << ( tag ) ) & ( ( 1u << MBEDTLS_ASN1_BMP_STRING ) | \ - ( 1u << MBEDTLS_ASN1_UTF8_STRING ) | \ - ( 1u << MBEDTLS_ASN1_T61_STRING ) | \ - ( 1u << MBEDTLS_ASN1_IA5_STRING ) | \ - ( 1u << MBEDTLS_ASN1_UNIVERSAL_STRING ) | \ - ( 1u << MBEDTLS_ASN1_PRINTABLE_STRING ) | \ - ( 1u << MBEDTLS_ASN1_BIT_STRING ) ) ) != 0 ) ) - -/* - * Bit masks for each of the components of an ASN.1 tag as specified in - * ITU X.690 (08/2015), section 8.1 "General rules for encoding", - * paragraph 8.1.2.2: - * - * Bit 8 7 6 5 1 - * +-------+-----+------------+ - * | Class | P/C | Tag number | - * +-------+-----+------------+ - */ -#define MBEDTLS_ASN1_TAG_CLASS_MASK 0xC0 -#define MBEDTLS_ASN1_TAG_PC_MASK 0x20 -#define MBEDTLS_ASN1_TAG_VALUE_MASK 0x1F - -/* \} name */ -/* \} addtogroup asn1_module */ - -/** Returns the size of the binary string, without the trailing \\0 */ -#define MBEDTLS_OID_SIZE(x) (sizeof(x) - 1) - -/** - * Compares an mbedtls_asn1_buf structure to a reference OID. - * - * Only works for 'defined' oid_str values (MBEDTLS_OID_HMAC_SHA1), you cannot use a - * 'unsigned char *oid' here! - */ -#define MBEDTLS_OID_CMP(oid_str, oid_buf) \ - ( ( MBEDTLS_OID_SIZE(oid_str) != (oid_buf)->len ) || \ - memcmp( (oid_str), (oid_buf)->p, (oid_buf)->len) != 0 ) - -#define MBEDTLS_OID_CMP_RAW(oid_str, oid_buf, oid_buf_len) \ - ( ( MBEDTLS_OID_SIZE(oid_str) != (oid_buf_len) ) || \ - memcmp( (oid_str), (oid_buf), (oid_buf_len) ) != 0 ) - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \name Functions to parse ASN.1 data structures - * \{ - */ - -/** - * Type-length-value structure that allows for ASN1 using DER. - */ -typedef struct mbedtls_asn1_buf -{ - int MBEDTLS_PRIVATE(tag); /**< ASN1 type, e.g. MBEDTLS_ASN1_UTF8_STRING. */ - size_t MBEDTLS_PRIVATE(len); /**< ASN1 length, in octets. */ - unsigned char *MBEDTLS_PRIVATE(p); /**< ASN1 data, e.g. in ASCII. */ -} -mbedtls_asn1_buf; - -/** - * Container for ASN1 bit strings. - */ -typedef struct mbedtls_asn1_bitstring -{ - size_t MBEDTLS_PRIVATE(len); /**< ASN1 length, in octets. */ - unsigned char MBEDTLS_PRIVATE(unused_bits); /**< Number of unused bits at the end of the string */ - unsigned char *MBEDTLS_PRIVATE(p); /**< Raw ASN1 data for the bit string */ -} -mbedtls_asn1_bitstring; - -/** - * Container for a sequence of ASN.1 items - */ -typedef struct mbedtls_asn1_sequence -{ - mbedtls_asn1_buf MBEDTLS_PRIVATE(buf); /**< Buffer containing the given ASN.1 item. */ - struct mbedtls_asn1_sequence *MBEDTLS_PRIVATE(next); /**< The next entry in the sequence. */ -} -mbedtls_asn1_sequence; - -/** - * Container for a sequence or list of 'named' ASN.1 data items - */ -typedef struct mbedtls_asn1_named_data -{ - mbedtls_asn1_buf MBEDTLS_PRIVATE(oid); /**< The object identifier. */ - mbedtls_asn1_buf MBEDTLS_PRIVATE(val); /**< The named value. */ - struct mbedtls_asn1_named_data *MBEDTLS_PRIVATE(next); /**< The next entry in the sequence. */ - unsigned char MBEDTLS_PRIVATE(next_merged); /**< Merge next item into the current one? */ -} -mbedtls_asn1_named_data; - -/** - * \brief Get the length of an ASN.1 element. - * Updates the pointer to immediately behind the length. - * - * \param p On entry, \c *p points to the first byte of the length, - * i.e. immediately after the tag. - * On successful completion, \c *p points to the first byte - * after the length, i.e. the first byte of the content. - * On error, the value of \c *p is undefined. - * \param end End of data. - * \param len On successful completion, \c *len contains the length - * read from the ASN.1 input. - * - * \return 0 if successful. - * \return #MBEDTLS_ERR_ASN1_OUT_OF_DATA if the ASN.1 element - * would end beyond \p end. - * \return #MBEDTLS_ERR_ASN1_INVALID_LENGTH if the length is unparseable. - */ -int mbedtls_asn1_get_len( unsigned char **p, - const unsigned char *end, - size_t *len ); - -/** - * \brief Get the tag and length of the element. - * Check for the requested tag. - * Updates the pointer to immediately behind the tag and length. - * - * \param p On entry, \c *p points to the start of the ASN.1 element. - * On successful completion, \c *p points to the first byte - * after the length, i.e. the first byte of the content. - * On error, the value of \c *p is undefined. - * \param end End of data. - * \param len On successful completion, \c *len contains the length - * read from the ASN.1 input. - * \param tag The expected tag. - * - * \return 0 if successful. - * \return #MBEDTLS_ERR_ASN1_UNEXPECTED_TAG if the data does not start - * with the requested tag. - * \return #MBEDTLS_ERR_ASN1_OUT_OF_DATA if the ASN.1 element - * would end beyond \p end. - * \return #MBEDTLS_ERR_ASN1_INVALID_LENGTH if the length is unparseable. - */ -int mbedtls_asn1_get_tag( unsigned char **p, - const unsigned char *end, - size_t *len, int tag ); - -/** - * \brief Retrieve a boolean ASN.1 tag and its value. - * Updates the pointer to immediately behind the full tag. - * - * \param p On entry, \c *p points to the start of the ASN.1 element. - * On successful completion, \c *p points to the first byte - * beyond the ASN.1 element. - * On error, the value of \c *p is undefined. - * \param end End of data. - * \param val On success, the parsed value (\c 0 or \c 1). - * - * \return 0 if successful. - * \return An ASN.1 error code if the input does not start with - * a valid ASN.1 BOOLEAN. - */ -int mbedtls_asn1_get_bool( unsigned char **p, - const unsigned char *end, - int *val ); - -/** - * \brief Retrieve an integer ASN.1 tag and its value. - * Updates the pointer to immediately behind the full tag. - * - * \param p On entry, \c *p points to the start of the ASN.1 element. - * On successful completion, \c *p points to the first byte - * beyond the ASN.1 element. - * On error, the value of \c *p is undefined. - * \param end End of data. - * \param val On success, the parsed value. - * - * \return 0 if successful. - * \return An ASN.1 error code if the input does not start with - * a valid ASN.1 INTEGER. - * \return #MBEDTLS_ERR_ASN1_INVALID_LENGTH if the parsed value does - * not fit in an \c int. - */ -int mbedtls_asn1_get_int( unsigned char **p, - const unsigned char *end, - int *val ); - -/** - * \brief Retrieve an enumerated ASN.1 tag and its value. - * Updates the pointer to immediately behind the full tag. - * - * \param p On entry, \c *p points to the start of the ASN.1 element. - * On successful completion, \c *p points to the first byte - * beyond the ASN.1 element. - * On error, the value of \c *p is undefined. - * \param end End of data. - * \param val On success, the parsed value. - * - * \return 0 if successful. - * \return An ASN.1 error code if the input does not start with - * a valid ASN.1 ENUMERATED. - * \return #MBEDTLS_ERR_ASN1_INVALID_LENGTH if the parsed value does - * not fit in an \c int. - */ -int mbedtls_asn1_get_enum( unsigned char **p, - const unsigned char *end, - int *val ); - -/** - * \brief Retrieve a bitstring ASN.1 tag and its value. - * Updates the pointer to immediately behind the full tag. - * - * \param p On entry, \c *p points to the start of the ASN.1 element. - * On successful completion, \c *p is equal to \p end. - * On error, the value of \c *p is undefined. - * \param end End of data. - * \param bs On success, ::mbedtls_asn1_bitstring information about - * the parsed value. - * - * \return 0 if successful. - * \return #MBEDTLS_ERR_ASN1_LENGTH_MISMATCH if the input contains - * extra data after a valid BIT STRING. - * \return An ASN.1 error code if the input does not start with - * a valid ASN.1 BIT STRING. - */ -int mbedtls_asn1_get_bitstring( unsigned char **p, const unsigned char *end, - mbedtls_asn1_bitstring *bs ); - -/** - * \brief Retrieve a bitstring ASN.1 tag without unused bits and its - * value. - * Updates the pointer to the beginning of the bit/octet string. - * - * \param p On entry, \c *p points to the start of the ASN.1 element. - * On successful completion, \c *p points to the first byte - * of the content of the BIT STRING. - * On error, the value of \c *p is undefined. - * \param end End of data. - * \param len On success, \c *len is the length of the content in bytes. - * - * \return 0 if successful. - * \return #MBEDTLS_ERR_ASN1_INVALID_DATA if the input starts with - * a valid BIT STRING with a nonzero number of unused bits. - * \return An ASN.1 error code if the input does not start with - * a valid ASN.1 BIT STRING. - */ -int mbedtls_asn1_get_bitstring_null( unsigned char **p, - const unsigned char *end, - size_t *len ); - -/** - * \brief Parses and splits an ASN.1 "SEQUENCE OF ". - * Updates the pointer to immediately behind the full sequence tag. - * - * This function allocates memory for the sequence elements. You can free - * the allocated memory with mbedtls_asn1_sequence_free(). - * - * \note On error, this function may return a partial list in \p cur. - * You must set `cur->next = NULL` before calling this function! - * Otherwise it is impossible to distinguish a previously non-null - * pointer from a pointer to an object allocated by this function. - * - * \note If the sequence is empty, this function does not modify - * \c *cur. If the sequence is valid and non-empty, this - * function sets `cur->buf.tag` to \p tag. This allows - * callers to distinguish between an empty sequence and - * a one-element sequence. - * - * \param p On entry, \c *p points to the start of the ASN.1 element. - * On successful completion, \c *p is equal to \p end. - * On error, the value of \c *p is undefined. - * \param end End of data. - * \param cur A ::mbedtls_asn1_sequence which this function fills. - * When this function returns, \c *cur is the head of a linked - * list. Each node in this list is allocated with - * mbedtls_calloc() apart from \p cur itself, and should - * therefore be freed with mbedtls_free(). - * The list describes the content of the sequence. - * The head of the list (i.e. \c *cur itself) describes the - * first element, `*cur->next` describes the second element, etc. - * For each element, `buf.tag == tag`, `buf.len` is the length - * of the content of the content of the element, and `buf.p` - * points to the first byte of the content (i.e. immediately - * past the length of the element). - * Note that list elements may be allocated even on error. - * \param tag Each element of the sequence must have this tag. - * - * \return 0 if successful. - * \return #MBEDTLS_ERR_ASN1_LENGTH_MISMATCH if the input contains - * extra data after a valid SEQUENCE OF \p tag. - * \return #MBEDTLS_ERR_ASN1_UNEXPECTED_TAG if the input starts with - * an ASN.1 SEQUENCE in which an element has a tag that - * is different from \p tag. - * \return #MBEDTLS_ERR_ASN1_ALLOC_FAILED if a memory allocation failed. - * \return An ASN.1 error code if the input does not start with - * a valid ASN.1 SEQUENCE. - */ -int mbedtls_asn1_get_sequence_of( unsigned char **p, - const unsigned char *end, - mbedtls_asn1_sequence *cur, - int tag ); -/** - * \brief Free a heap-allocated linked list presentation of - * an ASN.1 sequence, including the first element. - * - * There are two common ways to manage the memory used for the representation - * of a parsed ASN.1 sequence: - * - Allocate a head node `mbedtls_asn1_sequence *head` with mbedtls_calloc(). - * Pass this node as the `cur` argument to mbedtls_asn1_get_sequence_of(). - * When you have finished processing the sequence, - * call mbedtls_asn1_sequence_free() on `head`. - * - Allocate a head node `mbedtls_asn1_sequence *head` in any manner, - * for example on the stack. Make sure that `head->next == NULL`. - * Pass `head` as the `cur` argument to mbedtls_asn1_get_sequence_of(). - * When you have finished processing the sequence, - * call mbedtls_asn1_sequence_free() on `head->cur`, - * then free `head` itself in the appropriate manner. - * - * \param seq The address of the first sequence component. This may - * be \c NULL, in which case this functions returns - * immediately. - */ -void mbedtls_asn1_sequence_free( mbedtls_asn1_sequence *seq ); - -/** - * \brief Traverse an ASN.1 SEQUENCE container and - * call a callback for each entry. - * - * This function checks that the input is a SEQUENCE of elements that - * each have a "must" tag, and calls a callback function on the elements - * that have a "may" tag. - * - * For example, to validate that the input is a SEQUENCE of `tag1` and call - * `cb` on each element, use - * ``` - * mbedtls_asn1_traverse_sequence_of(&p, end, 0xff, tag1, 0, 0, cb, ctx); - * ``` - * - * To validate that the input is a SEQUENCE of ANY and call `cb` on - * each element, use - * ``` - * mbedtls_asn1_traverse_sequence_of(&p, end, 0, 0, 0, 0, cb, ctx); - * ``` - * - * To validate that the input is a SEQUENCE of CHOICE {NULL, OCTET STRING} - * and call `cb` on each element that is an OCTET STRING, use - * ``` - * mbedtls_asn1_traverse_sequence_of(&p, end, 0xfe, 0x04, 0xff, 0x04, cb, ctx); - * ``` - * - * The callback is called on the elements with a "may" tag from left to - * right. If the input is not a valid SEQUENCE of elements with a "must" tag, - * the callback is called on the elements up to the leftmost point where - * the input is invalid. - * - * \warning This function is still experimental and may change - * at any time. - * - * \param p The address of the pointer to the beginning of - * the ASN.1 SEQUENCE header. This is updated to - * point to the end of the ASN.1 SEQUENCE container - * on a successful invocation. - * \param end The end of the ASN.1 SEQUENCE container. - * \param tag_must_mask A mask to be applied to the ASN.1 tags found within - * the SEQUENCE before comparing to \p tag_must_value. - * \param tag_must_val The required value of each ASN.1 tag found in the - * SEQUENCE, after masking with \p tag_must_mask. - * Mismatching tags lead to an error. - * For example, a value of \c 0 for both \p tag_must_mask - * and \p tag_must_val means that every tag is allowed, - * while a value of \c 0xFF for \p tag_must_mask means - * that \p tag_must_val is the only allowed tag. - * \param tag_may_mask A mask to be applied to the ASN.1 tags found within - * the SEQUENCE before comparing to \p tag_may_value. - * \param tag_may_val The desired value of each ASN.1 tag found in the - * SEQUENCE, after masking with \p tag_may_mask. - * Mismatching tags will be silently ignored. - * For example, a value of \c 0 for \p tag_may_mask and - * \p tag_may_val means that any tag will be considered, - * while a value of \c 0xFF for \p tag_may_mask means - * that all tags with value different from \p tag_may_val - * will be ignored. - * \param cb The callback to trigger for each component - * in the ASN.1 SEQUENCE that matches \p tag_may_val. - * The callback function is called with the following - * parameters: - * - \p ctx. - * - The tag of the current element. - * - A pointer to the start of the current element's - * content inside the input. - * - The length of the content of the current element. - * If the callback returns a non-zero value, - * the function stops immediately, - * forwarding the callback's return value. - * \param ctx The context to be passed to the callback \p cb. - * - * \return \c 0 if successful the entire ASN.1 SEQUENCE - * was traversed without parsing or callback errors. - * \return #MBEDTLS_ERR_ASN1_LENGTH_MISMATCH if the input - * contains extra data after a valid SEQUENCE - * of elements with an accepted tag. - * \return #MBEDTLS_ERR_ASN1_UNEXPECTED_TAG if the input starts - * with an ASN.1 SEQUENCE in which an element has a tag - * that is not accepted. - * \return An ASN.1 error code if the input does not start with - * a valid ASN.1 SEQUENCE. - * \return A non-zero error code forwarded from the callback - * \p cb in case the latter returns a non-zero value. - */ -int mbedtls_asn1_traverse_sequence_of( - unsigned char **p, - const unsigned char *end, - unsigned char tag_must_mask, unsigned char tag_must_val, - unsigned char tag_may_mask, unsigned char tag_may_val, - int (*cb)( void *ctx, int tag, - unsigned char* start, size_t len ), - void *ctx ); - -#if defined(MBEDTLS_BIGNUM_C) -/** - * \brief Retrieve an integer ASN.1 tag and its value. - * Updates the pointer to immediately behind the full tag. - * - * \param p On entry, \c *p points to the start of the ASN.1 element. - * On successful completion, \c *p points to the first byte - * beyond the ASN.1 element. - * On error, the value of \c *p is undefined. - * \param end End of data. - * \param X On success, the parsed value. - * - * \return 0 if successful. - * \return An ASN.1 error code if the input does not start with - * a valid ASN.1 INTEGER. - * \return #MBEDTLS_ERR_ASN1_INVALID_LENGTH if the parsed value does - * not fit in an \c int. - * \return An MPI error code if the parsed value is too large. - */ -int mbedtls_asn1_get_mpi( unsigned char **p, - const unsigned char *end, - mbedtls_mpi *X ); -#endif /* MBEDTLS_BIGNUM_C */ - -/** - * \brief Retrieve an AlgorithmIdentifier ASN.1 sequence. - * Updates the pointer to immediately behind the full - * AlgorithmIdentifier. - * - * \param p On entry, \c *p points to the start of the ASN.1 element. - * On successful completion, \c *p points to the first byte - * beyond the AlgorithmIdentifier element. - * On error, the value of \c *p is undefined. - * \param end End of data. - * \param alg The buffer to receive the OID. - * \param params The buffer to receive the parameters. - * This is zeroized if there are no parameters. - * - * \return 0 if successful or a specific ASN.1 or MPI error code. - */ -int mbedtls_asn1_get_alg( unsigned char **p, - const unsigned char *end, - mbedtls_asn1_buf *alg, mbedtls_asn1_buf *params ); - -/** - * \brief Retrieve an AlgorithmIdentifier ASN.1 sequence with NULL or no - * params. - * Updates the pointer to immediately behind the full - * AlgorithmIdentifier. - * - * \param p On entry, \c *p points to the start of the ASN.1 element. - * On successful completion, \c *p points to the first byte - * beyond the AlgorithmIdentifier element. - * On error, the value of \c *p is undefined. - * \param end End of data. - * \param alg The buffer to receive the OID. - * - * \return 0 if successful or a specific ASN.1 or MPI error code. - */ -int mbedtls_asn1_get_alg_null( unsigned char **p, - const unsigned char *end, - mbedtls_asn1_buf *alg ); - -/** - * \brief Find a specific named_data entry in a sequence or list based on - * the OID. - * - * \param list The list to seek through - * \param oid The OID to look for - * \param len Size of the OID - * - * \return NULL if not found, or a pointer to the existing entry. - */ -const mbedtls_asn1_named_data *mbedtls_asn1_find_named_data( const mbedtls_asn1_named_data *list, - const char *oid, size_t len ); - -/** - * \brief Free a mbedtls_asn1_named_data entry - * - * \param entry The named data entry to free. - * This function calls mbedtls_free() on - * `entry->oid.p` and `entry->val.p`. - */ -void mbedtls_asn1_free_named_data( mbedtls_asn1_named_data *entry ); - -/** - * \brief Free all entries in a mbedtls_asn1_named_data list. - * - * \param head Pointer to the head of the list of named data entries to free. - * This function calls mbedtls_asn1_free_named_data() and - * mbedtls_free() on each list element and - * sets \c *head to \c NULL. - */ -void mbedtls_asn1_free_named_data_list( mbedtls_asn1_named_data **head ); - -#ifdef __cplusplus -} -#endif - -#endif /* asn1.h */ diff --git a/windows/mbedtls/asn1write.h b/windows/mbedtls/asn1write.h deleted file mode 100644 index d60ca0da..00000000 --- a/windows/mbedtls/asn1write.h +++ /dev/null @@ -1,366 +0,0 @@ -/** - * \file asn1write.h - * - * \brief ASN.1 buffer writing functionality - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_ASN1_WRITE_H -#define MBEDTLS_ASN1_WRITE_H - -#include "mbedtls/build_info.h" - -#include "mbedtls/asn1.h" - -#define MBEDTLS_ASN1_CHK_ADD(g, f) \ - do \ - { \ - if( ( ret = (f) ) < 0 ) \ - return( ret ); \ - else \ - (g) += ret; \ - } while( 0 ) - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief Write a length field in ASN.1 format. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param len The length value to write. - * - * \return The number of bytes written to \p p on success. - * \return A negative \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_asn1_write_len( unsigned char **p, const unsigned char *start, - size_t len ); -/** - * \brief Write an ASN.1 tag in ASN.1 format. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param tag The tag to write. - * - * \return The number of bytes written to \p p on success. - * \return A negative \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_asn1_write_tag( unsigned char **p, const unsigned char *start, - unsigned char tag ); - -/** - * \brief Write raw buffer data. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param buf The data buffer to write. - * \param size The length of the data buffer. - * - * \return The number of bytes written to \p p on success. - * \return A negative \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_asn1_write_raw_buffer( unsigned char **p, const unsigned char *start, - const unsigned char *buf, size_t size ); - -#if defined(MBEDTLS_BIGNUM_C) -/** - * \brief Write a arbitrary-precision number (#MBEDTLS_ASN1_INTEGER) - * in ASN.1 format. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param X The MPI to write. - * It must be non-negative. - * - * \return The number of bytes written to \p p on success. - * \return A negative \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_asn1_write_mpi( unsigned char **p, const unsigned char *start, - const mbedtls_mpi *X ); -#endif /* MBEDTLS_BIGNUM_C */ - -/** - * \brief Write a NULL tag (#MBEDTLS_ASN1_NULL) with zero data - * in ASN.1 format. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * - * \return The number of bytes written to \p p on success. - * \return A negative \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_asn1_write_null( unsigned char **p, const unsigned char *start ); - -/** - * \brief Write an OID tag (#MBEDTLS_ASN1_OID) and data - * in ASN.1 format. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param oid The OID to write. - * \param oid_len The length of the OID. - * - * \return The number of bytes written to \p p on success. - * \return A negative \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_asn1_write_oid( unsigned char **p, const unsigned char *start, - const char *oid, size_t oid_len ); - -/** - * \brief Write an AlgorithmIdentifier sequence in ASN.1 format. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param oid The OID of the algorithm to write. - * \param oid_len The length of the algorithm's OID. - * \param par_len The length of the parameters, which must be already written. - * If 0, NULL parameters are added - * - * \return The number of bytes written to \p p on success. - * \return A negative \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_asn1_write_algorithm_identifier( unsigned char **p, - const unsigned char *start, - const char *oid, size_t oid_len, - size_t par_len ); - -/** - * \brief Write a boolean tag (#MBEDTLS_ASN1_BOOLEAN) and value - * in ASN.1 format. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param boolean The boolean value to write, either \c 0 or \c 1. - * - * \return The number of bytes written to \p p on success. - * \return A negative \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_asn1_write_bool( unsigned char **p, const unsigned char *start, - int boolean ); - -/** - * \brief Write an int tag (#MBEDTLS_ASN1_INTEGER) and value - * in ASN.1 format. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param val The integer value to write. - * It must be non-negative. - * - * \return The number of bytes written to \p p on success. - * \return A negative \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_asn1_write_int( unsigned char **p, const unsigned char *start, int val ); - -/** - * \brief Write an enum tag (#MBEDTLS_ASN1_ENUMERATED) and value - * in ASN.1 format. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param val The integer value to write. - * - * \return The number of bytes written to \p p on success. - * \return A negative \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_asn1_write_enum( unsigned char **p, const unsigned char *start, int val ); - -/** - * \brief Write a string in ASN.1 format using a specific - * string encoding tag. - - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param tag The string encoding tag to write, e.g. - * #MBEDTLS_ASN1_UTF8_STRING. - * \param text The string to write. - * \param text_len The length of \p text in bytes (which might - * be strictly larger than the number of characters). - * - * \return The number of bytes written to \p p on success. - * \return A negative error code on failure. - */ -int mbedtls_asn1_write_tagged_string( unsigned char **p, const unsigned char *start, - int tag, const char *text, - size_t text_len ); - -/** - * \brief Write a string in ASN.1 format using the PrintableString - * string encoding tag (#MBEDTLS_ASN1_PRINTABLE_STRING). - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param text The string to write. - * \param text_len The length of \p text in bytes (which might - * be strictly larger than the number of characters). - * - * \return The number of bytes written to \p p on success. - * \return A negative error code on failure. - */ -int mbedtls_asn1_write_printable_string( unsigned char **p, - const unsigned char *start, - const char *text, size_t text_len ); - -/** - * \brief Write a UTF8 string in ASN.1 format using the UTF8String - * string encoding tag (#MBEDTLS_ASN1_UTF8_STRING). - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param text The string to write. - * \param text_len The length of \p text in bytes (which might - * be strictly larger than the number of characters). - * - * \return The number of bytes written to \p p on success. - * \return A negative error code on failure. - */ -int mbedtls_asn1_write_utf8_string( unsigned char **p, const unsigned char *start, - const char *text, size_t text_len ); - -/** - * \brief Write a string in ASN.1 format using the IA5String - * string encoding tag (#MBEDTLS_ASN1_IA5_STRING). - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param text The string to write. - * \param text_len The length of \p text in bytes (which might - * be strictly larger than the number of characters). - * - * \return The number of bytes written to \p p on success. - * \return A negative error code on failure. - */ -int mbedtls_asn1_write_ia5_string( unsigned char **p, const unsigned char *start, - const char *text, size_t text_len ); - -/** - * \brief Write a bitstring tag (#MBEDTLS_ASN1_BIT_STRING) and - * value in ASN.1 format. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param buf The bitstring to write. - * \param bits The total number of bits in the bitstring. - * - * \return The number of bytes written to \p p on success. - * \return A negative error code on failure. - */ -int mbedtls_asn1_write_bitstring( unsigned char **p, const unsigned char *start, - const unsigned char *buf, size_t bits ); - -/** - * \brief This function writes a named bitstring tag - * (#MBEDTLS_ASN1_BIT_STRING) and value in ASN.1 format. - * - * As stated in RFC 5280 Appendix B, trailing zeroes are - * omitted when encoding named bitstrings in DER. - * - * \note This function works backwards within the data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer which is used for bounds-checking. - * \param buf The bitstring to write. - * \param bits The total number of bits in the bitstring. - * - * \return The number of bytes written to \p p on success. - * \return A negative error code on failure. - */ -int mbedtls_asn1_write_named_bitstring( unsigned char **p, - const unsigned char *start, - const unsigned char *buf, - size_t bits ); - -/** - * \brief Write an octet string tag (#MBEDTLS_ASN1_OCTET_STRING) - * and value in ASN.1 format. - * - * \note This function works backwards in data buffer. - * - * \param p The reference to the current position pointer. - * \param start The start of the buffer, for bounds-checking. - * \param buf The buffer holding the data to write. - * \param size The length of the data buffer \p buf. - * - * \return The number of bytes written to \p p on success. - * \return A negative error code on failure. - */ -int mbedtls_asn1_write_octet_string( unsigned char **p, const unsigned char *start, - const unsigned char *buf, size_t size ); - -/** - * \brief Create or find a specific named_data entry for writing in a - * sequence or list based on the OID. If not already in there, - * a new entry is added to the head of the list. - * Warning: Destructive behaviour for the val data! - * - * \param list The pointer to the location of the head of the list to seek - * through (will be updated in case of a new entry). - * \param oid The OID to look for. - * \param oid_len The size of the OID. - * \param val The associated data to store. If this is \c NULL, - * no data is copied to the new or existing buffer. - * \param val_len The minimum length of the data buffer needed. - * If this is 0, do not allocate a buffer for the associated - * data. - * If the OID was already present, enlarge, shrink or free - * the existing buffer to fit \p val_len. - * - * \return A pointer to the new / existing entry on success. - * \return \c NULL if if there was a memory allocation error. - */ -mbedtls_asn1_named_data *mbedtls_asn1_store_named_data( mbedtls_asn1_named_data **list, - const char *oid, size_t oid_len, - const unsigned char *val, - size_t val_len ); - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_ASN1_WRITE_H */ diff --git a/windows/mbedtls/base64.h b/windows/mbedtls/base64.h deleted file mode 100644 index e3938b40..00000000 --- a/windows/mbedtls/base64.h +++ /dev/null @@ -1,92 +0,0 @@ -/** - * \file base64.h - * - * \brief RFC 1521 base64 encoding/decoding - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_BASE64_H -#define MBEDTLS_BASE64_H - -#include "mbedtls/build_info.h" - -#include - -#define MBEDTLS_ERR_BASE64_BUFFER_TOO_SMALL -0x002A /**< Output buffer too small. */ -#define MBEDTLS_ERR_BASE64_INVALID_CHARACTER -0x002C /**< Invalid character in input. */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief Encode a buffer into base64 format - * - * \param dst destination buffer - * \param dlen size of the destination buffer - * \param olen number of bytes written - * \param src source buffer - * \param slen amount of data to be encoded - * - * \return 0 if successful, or MBEDTLS_ERR_BASE64_BUFFER_TOO_SMALL. - * *olen is always updated to reflect the amount - * of data that has (or would have) been written. - * If that length cannot be represented, then no data is - * written to the buffer and *olen is set to the maximum - * length representable as a size_t. - * - * \note Call this function with dlen = 0 to obtain the - * required buffer size in *olen - */ -int mbedtls_base64_encode( unsigned char *dst, size_t dlen, size_t *olen, - const unsigned char *src, size_t slen ); - -/** - * \brief Decode a base64-formatted buffer - * - * \param dst destination buffer (can be NULL for checking size) - * \param dlen size of the destination buffer - * \param olen number of bytes written - * \param src source buffer - * \param slen amount of data to be decoded - * - * \return 0 if successful, MBEDTLS_ERR_BASE64_BUFFER_TOO_SMALL, or - * MBEDTLS_ERR_BASE64_INVALID_CHARACTER if the input data is - * not correct. *olen is always updated to reflect the amount - * of data that has (or would have) been written. - * - * \note Call this function with *dst = NULL or dlen = 0 to obtain - * the required buffer size in *olen - */ -int mbedtls_base64_decode( unsigned char *dst, size_t dlen, size_t *olen, - const unsigned char *src, size_t slen ); - -#if defined(MBEDTLS_SELF_TEST) -/** - * \brief Checkup routine - * - * \return 0 if successful, or 1 if the test failed - */ -int mbedtls_base64_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* base64.h */ diff --git a/windows/mbedtls/bignum.h b/windows/mbedtls/bignum.h deleted file mode 100644 index 5187d866..00000000 --- a/windows/mbedtls/bignum.h +++ /dev/null @@ -1,1021 +0,0 @@ -/** - * \file bignum.h - * - * \brief Multi-precision integer library - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_BIGNUM_H -#define MBEDTLS_BIGNUM_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -#if defined(MBEDTLS_FS_IO) -#include -#endif - -#define MBEDTLS_ERR_MPI_FILE_IO_ERROR -0x0002 /**< An error occurred while reading from or writing to a file. */ -#define MBEDTLS_ERR_MPI_BAD_INPUT_DATA -0x0004 /**< Bad input parameters to function. */ -#define MBEDTLS_ERR_MPI_INVALID_CHARACTER -0x0006 /**< There is an invalid character in the digit string. */ -#define MBEDTLS_ERR_MPI_BUFFER_TOO_SMALL -0x0008 /**< The buffer is too small to write to. */ -#define MBEDTLS_ERR_MPI_NEGATIVE_VALUE -0x000A /**< The input arguments are negative or result in illegal output. */ -#define MBEDTLS_ERR_MPI_DIVISION_BY_ZERO -0x000C /**< The input argument for division is zero, which is not allowed. */ -#define MBEDTLS_ERR_MPI_NOT_ACCEPTABLE -0x000E /**< The input arguments are not acceptable. */ -#define MBEDTLS_ERR_MPI_ALLOC_FAILED -0x0010 /**< Memory allocation failed. */ - -#define MBEDTLS_MPI_CHK(f) \ - do \ - { \ - if( ( ret = (f) ) != 0 ) \ - goto cleanup; \ - } while( 0 ) - -/* - * Maximum size MPIs are allowed to grow to in number of limbs. - */ -#define MBEDTLS_MPI_MAX_LIMBS 10000 - -#if !defined(MBEDTLS_MPI_WINDOW_SIZE) -/* - * Maximum window size used for modular exponentiation. Default: 6 - * Minimum value: 1. Maximum value: 6. - * - * Result is an array of ( 2 ** MBEDTLS_MPI_WINDOW_SIZE ) MPIs used - * for the sliding window calculation. (So 64 by default) - * - * Reduction in size, reduces speed. - */ -#define MBEDTLS_MPI_WINDOW_SIZE 6 /**< Maximum window size used. */ -#endif /* !MBEDTLS_MPI_WINDOW_SIZE */ - -#if !defined(MBEDTLS_MPI_MAX_SIZE) -/* - * Maximum size of MPIs allowed in bits and bytes for user-MPIs. - * ( Default: 512 bytes => 4096 bits, Maximum tested: 2048 bytes => 16384 bits ) - * - * Note: Calculations can temporarily result in larger MPIs. So the number - * of limbs required (MBEDTLS_MPI_MAX_LIMBS) is higher. - */ -#define MBEDTLS_MPI_MAX_SIZE 1024 /**< Maximum number of bytes for usable MPIs. */ -#endif /* !MBEDTLS_MPI_MAX_SIZE */ - -#define MBEDTLS_MPI_MAX_BITS ( 8 * MBEDTLS_MPI_MAX_SIZE ) /**< Maximum number of bits for usable MPIs. */ - -/* - * When reading from files with mbedtls_mpi_read_file() and writing to files with - * mbedtls_mpi_write_file() the buffer should have space - * for a (short) label, the MPI (in the provided radix), the newline - * characters and the '\0'. - * - * By default we assume at least a 10 char label, a minimum radix of 10 - * (decimal) and a maximum of 4096 bit numbers (1234 decimal chars). - * Autosized at compile time for at least a 10 char label, a minimum radix - * of 10 (decimal) for a number of MBEDTLS_MPI_MAX_BITS size. - * - * This used to be statically sized to 1250 for a maximum of 4096 bit - * numbers (1234 decimal chars). - * - * Calculate using the formula: - * MBEDTLS_MPI_RW_BUFFER_SIZE = ceil(MBEDTLS_MPI_MAX_BITS / ln(10) * ln(2)) + - * LabelSize + 6 - */ -#define MBEDTLS_MPI_MAX_BITS_SCALE100 ( 100 * MBEDTLS_MPI_MAX_BITS ) -#define MBEDTLS_LN_2_DIV_LN_10_SCALE100 332 -#define MBEDTLS_MPI_RW_BUFFER_SIZE ( ((MBEDTLS_MPI_MAX_BITS_SCALE100 + MBEDTLS_LN_2_DIV_LN_10_SCALE100 - 1) / MBEDTLS_LN_2_DIV_LN_10_SCALE100) + 10 + 6 ) - -/* - * Define the base integer type, architecture-wise. - * - * 32 or 64-bit integer types can be forced regardless of the underlying - * architecture by defining MBEDTLS_HAVE_INT32 or MBEDTLS_HAVE_INT64 - * respectively and undefining MBEDTLS_HAVE_ASM. - * - * Double-width integers (e.g. 128-bit in 64-bit architectures) can be - * disabled by defining MBEDTLS_NO_UDBL_DIVISION. - */ -#if !defined(MBEDTLS_HAVE_INT32) - #if defined(_MSC_VER) && defined(_M_AMD64) - /* Always choose 64-bit when using MSC */ - #if !defined(MBEDTLS_HAVE_INT64) - #define MBEDTLS_HAVE_INT64 - #endif /* !MBEDTLS_HAVE_INT64 */ - typedef int64_t mbedtls_mpi_sint; - typedef uint64_t mbedtls_mpi_uint; - #elif defined(__GNUC__) && ( \ - defined(__amd64__) || defined(__x86_64__) || \ - defined(__ppc64__) || defined(__powerpc64__) || \ - defined(__ia64__) || defined(__alpha__) || \ - ( defined(__sparc__) && defined(__arch64__) ) || \ - defined(__s390x__) || defined(__mips64) || \ - defined(__aarch64__) ) - #if !defined(MBEDTLS_HAVE_INT64) - #define MBEDTLS_HAVE_INT64 - #endif /* MBEDTLS_HAVE_INT64 */ - typedef int64_t mbedtls_mpi_sint; - typedef uint64_t mbedtls_mpi_uint; - #if !defined(MBEDTLS_NO_UDBL_DIVISION) - /* mbedtls_t_udbl defined as 128-bit unsigned int */ - typedef unsigned int mbedtls_t_udbl __attribute__((mode(TI))); - #define MBEDTLS_HAVE_UDBL - #endif /* !MBEDTLS_NO_UDBL_DIVISION */ - #elif defined(__ARMCC_VERSION) && defined(__aarch64__) - /* - * __ARMCC_VERSION is defined for both armcc and armclang and - * __aarch64__ is only defined by armclang when compiling 64-bit code - */ - #if !defined(MBEDTLS_HAVE_INT64) - #define MBEDTLS_HAVE_INT64 - #endif /* !MBEDTLS_HAVE_INT64 */ - typedef int64_t mbedtls_mpi_sint; - typedef uint64_t mbedtls_mpi_uint; - #if !defined(MBEDTLS_NO_UDBL_DIVISION) - /* mbedtls_t_udbl defined as 128-bit unsigned int */ - typedef __uint128_t mbedtls_t_udbl; - #define MBEDTLS_HAVE_UDBL - #endif /* !MBEDTLS_NO_UDBL_DIVISION */ - #elif defined(MBEDTLS_HAVE_INT64) - /* Force 64-bit integers with unknown compiler */ - typedef int64_t mbedtls_mpi_sint; - typedef uint64_t mbedtls_mpi_uint; - #endif -#endif /* !MBEDTLS_HAVE_INT32 */ - -#if !defined(MBEDTLS_HAVE_INT64) - /* Default to 32-bit compilation */ - #if !defined(MBEDTLS_HAVE_INT32) - #define MBEDTLS_HAVE_INT32 - #endif /* !MBEDTLS_HAVE_INT32 */ - typedef int32_t mbedtls_mpi_sint; - typedef uint32_t mbedtls_mpi_uint; - #if !defined(MBEDTLS_NO_UDBL_DIVISION) - typedef uint64_t mbedtls_t_udbl; - #define MBEDTLS_HAVE_UDBL - #endif /* !MBEDTLS_NO_UDBL_DIVISION */ -#endif /* !MBEDTLS_HAVE_INT64 */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief MPI structure - */ -typedef struct mbedtls_mpi -{ - int MBEDTLS_PRIVATE(s); /*!< Sign: -1 if the mpi is negative, 1 otherwise */ - size_t MBEDTLS_PRIVATE(n); /*!< total # of limbs */ - mbedtls_mpi_uint *MBEDTLS_PRIVATE(p); /*!< pointer to limbs */ -} -mbedtls_mpi; - -/** - * \brief Initialize an MPI context. - * - * This makes the MPI ready to be set or freed, - * but does not define a value for the MPI. - * - * \param X The MPI context to initialize. This must not be \c NULL. - */ -void mbedtls_mpi_init( mbedtls_mpi *X ); - -/** - * \brief This function frees the components of an MPI context. - * - * \param X The MPI context to be cleared. This may be \c NULL, - * in which case this function is a no-op. If it is - * not \c NULL, it must point to an initialized MPI. - */ -void mbedtls_mpi_free( mbedtls_mpi *X ); - -/** - * \brief Enlarge an MPI to the specified number of limbs. - * - * \note This function does nothing if the MPI is - * already large enough. - * - * \param X The MPI to grow. It must be initialized. - * \param nblimbs The target number of limbs. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_mpi_grow( mbedtls_mpi *X, size_t nblimbs ); - -/** - * \brief This function resizes an MPI downwards, keeping at least the - * specified number of limbs. - * - * If \c X is smaller than \c nblimbs, it is resized up - * instead. - * - * \param X The MPI to shrink. This must point to an initialized MPI. - * \param nblimbs The minimum number of limbs to keep. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed - * (this can only happen when resizing up). - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_mpi_shrink( mbedtls_mpi *X, size_t nblimbs ); - -/** - * \brief Make a copy of an MPI. - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param Y The source MPI. This must point to an initialized MPI. - * - * \note The limb-buffer in the destination MPI is enlarged - * if necessary to hold the value in the source MPI. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_mpi_copy( mbedtls_mpi *X, const mbedtls_mpi *Y ); - -/** - * \brief Swap the contents of two MPIs. - * - * \param X The first MPI. It must be initialized. - * \param Y The second MPI. It must be initialized. - */ -void mbedtls_mpi_swap( mbedtls_mpi *X, mbedtls_mpi *Y ); - -/** - * \brief Perform a safe conditional copy of MPI which doesn't - * reveal whether the condition was true or not. - * - * \param X The MPI to conditionally assign to. This must point - * to an initialized MPI. - * \param Y The MPI to be assigned from. This must point to an - * initialized MPI. - * \param assign The condition deciding whether to perform the - * assignment or not. Possible values: - * * \c 1: Perform the assignment `X = Y`. - * * \c 0: Keep the original value of \p X. - * - * \note This function is equivalent to - * `if( assign ) mbedtls_mpi_copy( X, Y );` - * except that it avoids leaking any information about whether - * the assignment was done or not (the above code may leak - * information through branch prediction and/or memory access - * patterns analysis). - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_mpi_safe_cond_assign( mbedtls_mpi *X, const mbedtls_mpi *Y, unsigned char assign ); - -/** - * \brief Perform a safe conditional swap which doesn't - * reveal whether the condition was true or not. - * - * \param X The first MPI. This must be initialized. - * \param Y The second MPI. This must be initialized. - * \param assign The condition deciding whether to perform - * the swap or not. Possible values: - * * \c 1: Swap the values of \p X and \p Y. - * * \c 0: Keep the original values of \p X and \p Y. - * - * \note This function is equivalent to - * if( assign ) mbedtls_mpi_swap( X, Y ); - * except that it avoids leaking any information about whether - * the assignment was done or not (the above code may leak - * information through branch prediction and/or memory access - * patterns analysis). - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed. - * \return Another negative error code on other kinds of failure. - * - */ -int mbedtls_mpi_safe_cond_swap( mbedtls_mpi *X, mbedtls_mpi *Y, unsigned char assign ); - -/** - * \brief Store integer value in MPI. - * - * \param X The MPI to set. This must be initialized. - * \param z The value to use. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_mpi_lset( mbedtls_mpi *X, mbedtls_mpi_sint z ); - -/** - * \brief Get a specific bit from an MPI. - * - * \param X The MPI to query. This must be initialized. - * \param pos Zero-based index of the bit to query. - * - * \return \c 0 or \c 1 on success, depending on whether bit \c pos - * of \c X is unset or set. - * \return A negative error code on failure. - */ -int mbedtls_mpi_get_bit( const mbedtls_mpi *X, size_t pos ); - -/** - * \brief Modify a specific bit in an MPI. - * - * \note This function will grow the target MPI if necessary to set a - * bit to \c 1 in a not yet existing limb. It will not grow if - * the bit should be set to \c 0. - * - * \param X The MPI to modify. This must be initialized. - * \param pos Zero-based index of the bit to modify. - * \param val The desired value of bit \c pos: \c 0 or \c 1. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_mpi_set_bit( mbedtls_mpi *X, size_t pos, unsigned char val ); - -/** - * \brief Return the number of bits of value \c 0 before the - * least significant bit of value \c 1. - * - * \note This is the same as the zero-based index of - * the least significant bit of value \c 1. - * - * \param X The MPI to query. - * - * \return The number of bits of value \c 0 before the least significant - * bit of value \c 1 in \p X. - */ -size_t mbedtls_mpi_lsb( const mbedtls_mpi *X ); - -/** - * \brief Return the number of bits up to and including the most - * significant bit of value \c 1. - * - * * \note This is same as the one-based index of the most - * significant bit of value \c 1. - * - * \param X The MPI to query. This must point to an initialized MPI. - * - * \return The number of bits up to and including the most - * significant bit of value \c 1. - */ -size_t mbedtls_mpi_bitlen( const mbedtls_mpi *X ); - -/** - * \brief Return the total size of an MPI value in bytes. - * - * \param X The MPI to use. This must point to an initialized MPI. - * - * \note The value returned by this function may be less than - * the number of bytes used to store \p X internally. - * This happens if and only if there are trailing bytes - * of value zero. - * - * \return The least number of bytes capable of storing - * the absolute value of \p X. - */ -size_t mbedtls_mpi_size( const mbedtls_mpi *X ); - -/** - * \brief Import an MPI from an ASCII string. - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param radix The numeric base of the input string. - * \param s Null-terminated string buffer. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_mpi_read_string( mbedtls_mpi *X, int radix, const char *s ); - -/** - * \brief Export an MPI to an ASCII string. - * - * \param X The source MPI. This must point to an initialized MPI. - * \param radix The numeric base of the output string. - * \param buf The buffer to write the string to. This must be writable - * buffer of length \p buflen Bytes. - * \param buflen The available size in Bytes of \p buf. - * \param olen The address at which to store the length of the string - * written, including the final \c NULL byte. This must - * not be \c NULL. - * - * \note You can call this function with `buflen == 0` to obtain the - * minimum required buffer size in `*olen`. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_BUFFER_TOO_SMALL if the target buffer \p buf - * is too small to hold the value of \p X in the desired base. - * In this case, `*olen` is nonetheless updated to contain the - * size of \p buf required for a successful call. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_write_string( const mbedtls_mpi *X, int radix, - char *buf, size_t buflen, size_t *olen ); - -#if defined(MBEDTLS_FS_IO) -/** - * \brief Read an MPI from a line in an opened file. - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param radix The numeric base of the string representation used - * in the source line. - * \param fin The input file handle to use. This must not be \c NULL. - * - * \note On success, this function advances the file stream - * to the end of the current line or to EOF. - * - * The function returns \c 0 on an empty line. - * - * Leading whitespaces are ignored, as is a - * '0x' prefix for radix \c 16. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_BUFFER_TOO_SMALL if the file read buffer - * is too small. - * \return Another negative error code on failure. - */ -int mbedtls_mpi_read_file( mbedtls_mpi *X, int radix, FILE *fin ); - -/** - * \brief Export an MPI into an opened file. - * - * \param p A string prefix to emit prior to the MPI data. - * For example, this might be a label, or "0x" when - * printing in base \c 16. This may be \c NULL if no prefix - * is needed. - * \param X The source MPI. This must point to an initialized MPI. - * \param radix The numeric base to be used in the emitted string. - * \param fout The output file handle. This may be \c NULL, in which case - * the output is written to \c stdout. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_mpi_write_file( const char *p, const mbedtls_mpi *X, - int radix, FILE *fout ); -#endif /* MBEDTLS_FS_IO */ - -/** - * \brief Import an MPI from unsigned big endian binary data. - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param buf The input buffer. This must be a readable buffer of length - * \p buflen Bytes. - * \param buflen The length of the input buffer \p p in Bytes. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_read_binary( mbedtls_mpi *X, const unsigned char *buf, - size_t buflen ); - -/** - * \brief Import X from unsigned binary data, little endian - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param buf The input buffer. This must be a readable buffer of length - * \p buflen Bytes. - * \param buflen The length of the input buffer \p p in Bytes. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_read_binary_le( mbedtls_mpi *X, - const unsigned char *buf, size_t buflen ); - -/** - * \brief Export X into unsigned binary data, big endian. - * Always fills the whole buffer, which will start with zeros - * if the number is smaller. - * - * \param X The source MPI. This must point to an initialized MPI. - * \param buf The output buffer. This must be a writable buffer of length - * \p buflen Bytes. - * \param buflen The size of the output buffer \p buf in Bytes. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_BUFFER_TOO_SMALL if \p buf isn't - * large enough to hold the value of \p X. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_write_binary( const mbedtls_mpi *X, unsigned char *buf, - size_t buflen ); - -/** - * \brief Export X into unsigned binary data, little endian. - * Always fills the whole buffer, which will end with zeros - * if the number is smaller. - * - * \param X The source MPI. This must point to an initialized MPI. - * \param buf The output buffer. This must be a writable buffer of length - * \p buflen Bytes. - * \param buflen The size of the output buffer \p buf in Bytes. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_BUFFER_TOO_SMALL if \p buf isn't - * large enough to hold the value of \p X. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_write_binary_le( const mbedtls_mpi *X, - unsigned char *buf, size_t buflen ); - -/** - * \brief Perform a left-shift on an MPI: X <<= count - * - * \param X The MPI to shift. This must point to an initialized MPI. - * \param count The number of bits to shift by. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_shift_l( mbedtls_mpi *X, size_t count ); - -/** - * \brief Perform a right-shift on an MPI: X >>= count - * - * \param X The MPI to shift. This must point to an initialized MPI. - * \param count The number of bits to shift by. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_shift_r( mbedtls_mpi *X, size_t count ); - -/** - * \brief Compare the absolute values of two MPIs. - * - * \param X The left-hand MPI. This must point to an initialized MPI. - * \param Y The right-hand MPI. This must point to an initialized MPI. - * - * \return \c 1 if `|X|` is greater than `|Y|`. - * \return \c -1 if `|X|` is lesser than `|Y|`. - * \return \c 0 if `|X|` is equal to `|Y|`. - */ -int mbedtls_mpi_cmp_abs( const mbedtls_mpi *X, const mbedtls_mpi *Y ); - -/** - * \brief Compare two MPIs. - * - * \param X The left-hand MPI. This must point to an initialized MPI. - * \param Y The right-hand MPI. This must point to an initialized MPI. - * - * \return \c 1 if \p X is greater than \p Y. - * \return \c -1 if \p X is lesser than \p Y. - * \return \c 0 if \p X is equal to \p Y. - */ -int mbedtls_mpi_cmp_mpi( const mbedtls_mpi *X, const mbedtls_mpi *Y ); - -/** - * \brief Check if an MPI is less than the other in constant time. - * - * \param X The left-hand MPI. This must point to an initialized MPI - * with the same allocated length as Y. - * \param Y The right-hand MPI. This must point to an initialized MPI - * with the same allocated length as X. - * \param ret The result of the comparison: - * \c 1 if \p X is less than \p Y. - * \c 0 if \p X is greater than or equal to \p Y. - * - * \return 0 on success. - * \return MBEDTLS_ERR_MPI_BAD_INPUT_DATA if the allocated length of - * the two input MPIs is not the same. - */ -int mbedtls_mpi_lt_mpi_ct( const mbedtls_mpi *X, const mbedtls_mpi *Y, - unsigned *ret ); - -/** - * \brief Compare an MPI with an integer. - * - * \param X The left-hand MPI. This must point to an initialized MPI. - * \param z The integer value to compare \p X to. - * - * \return \c 1 if \p X is greater than \p z. - * \return \c -1 if \p X is lesser than \p z. - * \return \c 0 if \p X is equal to \p z. - */ -int mbedtls_mpi_cmp_int( const mbedtls_mpi *X, mbedtls_mpi_sint z ); - -/** - * \brief Perform an unsigned addition of MPIs: X = |A| + |B| - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param A The first summand. This must point to an initialized MPI. - * \param B The second summand. This must point to an initialized MPI. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_add_abs( mbedtls_mpi *X, const mbedtls_mpi *A, - const mbedtls_mpi *B ); - -/** - * \brief Perform an unsigned subtraction of MPIs: X = |A| - |B| - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param A The minuend. This must point to an initialized MPI. - * \param B The subtrahend. This must point to an initialized MPI. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_NEGATIVE_VALUE if \p B is greater than \p A. - * \return Another negative error code on different kinds of failure. - * - */ -int mbedtls_mpi_sub_abs( mbedtls_mpi *X, const mbedtls_mpi *A, - const mbedtls_mpi *B ); - -/** - * \brief Perform a signed addition of MPIs: X = A + B - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param A The first summand. This must point to an initialized MPI. - * \param B The second summand. This must point to an initialized MPI. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_add_mpi( mbedtls_mpi *X, const mbedtls_mpi *A, - const mbedtls_mpi *B ); - -/** - * \brief Perform a signed subtraction of MPIs: X = A - B - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param A The minuend. This must point to an initialized MPI. - * \param B The subtrahend. This must point to an initialized MPI. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_sub_mpi( mbedtls_mpi *X, const mbedtls_mpi *A, - const mbedtls_mpi *B ); - -/** - * \brief Perform a signed addition of an MPI and an integer: X = A + b - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param A The first summand. This must point to an initialized MPI. - * \param b The second summand. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_add_int( mbedtls_mpi *X, const mbedtls_mpi *A, - mbedtls_mpi_sint b ); - -/** - * \brief Perform a signed subtraction of an MPI and an integer: - * X = A - b - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param A The minuend. This must point to an initialized MPI. - * \param b The subtrahend. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_sub_int( mbedtls_mpi *X, const mbedtls_mpi *A, - mbedtls_mpi_sint b ); - -/** - * \brief Perform a multiplication of two MPIs: X = A * B - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param A The first factor. This must point to an initialized MPI. - * \param B The second factor. This must point to an initialized MPI. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return Another negative error code on different kinds of failure. - * - */ -int mbedtls_mpi_mul_mpi( mbedtls_mpi *X, const mbedtls_mpi *A, - const mbedtls_mpi *B ); - -/** - * \brief Perform a multiplication of an MPI with an unsigned integer: - * X = A * b - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param A The first factor. This must point to an initialized MPI. - * \param b The second factor. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return Another negative error code on different kinds of failure. - * - */ -int mbedtls_mpi_mul_int( mbedtls_mpi *X, const mbedtls_mpi *A, - mbedtls_mpi_uint b ); - -/** - * \brief Perform a division with remainder of two MPIs: - * A = Q * B + R - * - * \param Q The destination MPI for the quotient. - * This may be \c NULL if the value of the - * quotient is not needed. - * \param R The destination MPI for the remainder value. - * This may be \c NULL if the value of the - * remainder is not needed. - * \param A The dividend. This must point to an initialized MPi. - * \param B The divisor. This must point to an initialized MPI. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed. - * \return #MBEDTLS_ERR_MPI_DIVISION_BY_ZERO if \p B equals zero. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_div_mpi( mbedtls_mpi *Q, mbedtls_mpi *R, const mbedtls_mpi *A, - const mbedtls_mpi *B ); - -/** - * \brief Perform a division with remainder of an MPI by an integer: - * A = Q * b + R - * - * \param Q The destination MPI for the quotient. - * This may be \c NULL if the value of the - * quotient is not needed. - * \param R The destination MPI for the remainder value. - * This may be \c NULL if the value of the - * remainder is not needed. - * \param A The dividend. This must point to an initialized MPi. - * \param b The divisor. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed. - * \return #MBEDTLS_ERR_MPI_DIVISION_BY_ZERO if \p b equals zero. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_div_int( mbedtls_mpi *Q, mbedtls_mpi *R, const mbedtls_mpi *A, - mbedtls_mpi_sint b ); - -/** - * \brief Perform a modular reduction. R = A mod B - * - * \param R The destination MPI for the residue value. - * This must point to an initialized MPI. - * \param A The MPI to compute the residue of. - * This must point to an initialized MPI. - * \param B The base of the modular reduction. - * This must point to an initialized MPI. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return #MBEDTLS_ERR_MPI_DIVISION_BY_ZERO if \p B equals zero. - * \return #MBEDTLS_ERR_MPI_NEGATIVE_VALUE if \p B is negative. - * \return Another negative error code on different kinds of failure. - * - */ -int mbedtls_mpi_mod_mpi( mbedtls_mpi *R, const mbedtls_mpi *A, - const mbedtls_mpi *B ); - -/** - * \brief Perform a modular reduction with respect to an integer. - * r = A mod b - * - * \param r The address at which to store the residue. - * This must not be \c NULL. - * \param A The MPI to compute the residue of. - * This must point to an initialized MPi. - * \param b The integer base of the modular reduction. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return #MBEDTLS_ERR_MPI_DIVISION_BY_ZERO if \p b equals zero. - * \return #MBEDTLS_ERR_MPI_NEGATIVE_VALUE if \p b is negative. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_mod_int( mbedtls_mpi_uint *r, const mbedtls_mpi *A, - mbedtls_mpi_sint b ); - -/** - * \brief Perform a sliding-window exponentiation: X = A^E mod N - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param A The base of the exponentiation. - * This must point to an initialized MPI. - * \param E The exponent MPI. This must point to an initialized MPI. - * \param N The base for the modular reduction. This must point to an - * initialized MPI. - * \param _RR A helper MPI depending solely on \p N which can be used to - * speed-up multiple modular exponentiations for the same value - * of \p N. This may be \c NULL. If it is not \c NULL, it must - * point to an initialized MPI. If it hasn't been used after - * the call to mbedtls_mpi_init(), this function will compute - * the helper value and store it in \p _RR for reuse on - * subsequent calls to this function. Otherwise, the function - * will assume that \p _RR holds the helper value set by a - * previous call to mbedtls_mpi_exp_mod(), and reuse it. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return #MBEDTLS_ERR_MPI_BAD_INPUT_DATA if \c N is negative or - * even, or if \c E is negative. - * \return Another negative error code on different kinds of failures. - * - */ -int mbedtls_mpi_exp_mod( mbedtls_mpi *X, const mbedtls_mpi *A, - const mbedtls_mpi *E, const mbedtls_mpi *N, - mbedtls_mpi *_RR ); - -/** - * \brief Fill an MPI with a number of random bytes. - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param size The number of random bytes to generate. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG parameter to be passed to \p f_rng. This may be - * \c NULL if \p f_rng doesn't need a context argument. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return Another negative error code on failure. - * - * \note The bytes obtained from the RNG are interpreted - * as a big-endian representation of an MPI; this can - * be relevant in applications like deterministic ECDSA. - */ -int mbedtls_mpi_fill_random( mbedtls_mpi *X, size_t size, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** Generate a random number uniformly in a range. - * - * This function generates a random number between \p min inclusive and - * \p N exclusive. - * - * The procedure complies with RFC 6979 §3.3 (deterministic ECDSA) - * when the RNG is a suitably parametrized instance of HMAC_DRBG - * and \p min is \c 1. - * - * \note There are `N - min` possible outputs. The lower bound - * \p min can be reached, but the upper bound \p N cannot. - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param min The minimum value to return. - * It must be nonnegative. - * \param N The upper bound of the range, exclusive. - * In other words, this is one plus the maximum value to return. - * \p N must be strictly larger than \p min. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG parameter to be passed to \p f_rng. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return #MBEDTLS_ERR_MPI_BAD_INPUT_DATA if \p min or \p N is invalid - * or if they are incompatible. - * \return #MBEDTLS_ERR_MPI_NOT_ACCEPTABLE if the implementation was - * unable to find a suitable value within a limited number - * of attempts. This has a negligible probability if \p N - * is significantly larger than \p min, which is the case - * for all usual cryptographic applications. - * \return Another negative error code on failure. - */ -int mbedtls_mpi_random( mbedtls_mpi *X, - mbedtls_mpi_sint min, - const mbedtls_mpi *N, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief Compute the greatest common divisor: G = gcd(A, B) - * - * \param G The destination MPI. This must point to an initialized MPI. - * \param A The first operand. This must point to an initialized MPI. - * \param B The second operand. This must point to an initialized MPI. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_mpi_gcd( mbedtls_mpi *G, const mbedtls_mpi *A, - const mbedtls_mpi *B ); - -/** - * \brief Compute the modular inverse: X = A^-1 mod N - * - * \param X The destination MPI. This must point to an initialized MPI. - * \param A The MPI to calculate the modular inverse of. This must point - * to an initialized MPI. - * \param N The base of the modular inversion. This must point to an - * initialized MPI. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return #MBEDTLS_ERR_MPI_BAD_INPUT_DATA if \p N is less than - * or equal to one. - * \return #MBEDTLS_ERR_MPI_NOT_ACCEPTABLE if \p has no modular inverse - * with respect to \p N. - */ -int mbedtls_mpi_inv_mod( mbedtls_mpi *X, const mbedtls_mpi *A, - const mbedtls_mpi *N ); - -/** - * \brief Miller-Rabin primality test. - * - * \warning If \p X is potentially generated by an adversary, for example - * when validating cryptographic parameters that you didn't - * generate yourself and that are supposed to be prime, then - * \p rounds should be at least the half of the security - * strength of the cryptographic algorithm. On the other hand, - * if \p X is chosen uniformly or non-adversially (as is the - * case when mbedtls_mpi_gen_prime calls this function), then - * \p rounds can be much lower. - * - * \param X The MPI to check for primality. - * This must point to an initialized MPI. - * \param rounds The number of bases to perform the Miller-Rabin primality - * test for. The probability of returning 0 on a composite is - * at most 2-2*\p rounds. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG parameter to be passed to \p f_rng. - * This may be \c NULL if \p f_rng doesn't use - * a context parameter. - * - * \return \c 0 if successful, i.e. \p X is probably prime. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return #MBEDTLS_ERR_MPI_NOT_ACCEPTABLE if \p X is not prime. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_mpi_is_prime_ext( const mbedtls_mpi *X, int rounds, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); -/** - * \brief Flags for mbedtls_mpi_gen_prime() - * - * Each of these flags is a constraint on the result X returned by - * mbedtls_mpi_gen_prime(). - */ -typedef enum { - MBEDTLS_MPI_GEN_PRIME_FLAG_DH = 0x0001, /**< (X-1)/2 is prime too */ - MBEDTLS_MPI_GEN_PRIME_FLAG_LOW_ERR = 0x0002, /**< lower error rate from 2-80 to 2-128 */ -} mbedtls_mpi_gen_prime_flag_t; - -/** - * \brief Generate a prime number. - * - * \param X The destination MPI to store the generated prime in. - * This must point to an initialized MPi. - * \param nbits The required size of the destination MPI in bits. - * This must be between \c 3 and #MBEDTLS_MPI_MAX_BITS. - * \param flags A mask of flags of type #mbedtls_mpi_gen_prime_flag_t. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG parameter to be passed to \p f_rng. - * This may be \c NULL if \p f_rng doesn't use - * a context parameter. - * - * \return \c 0 if successful, in which case \p X holds a - * probably prime number. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if a memory allocation failed. - * \return #MBEDTLS_ERR_MPI_BAD_INPUT_DATA if `nbits` is not between - * \c 3 and #MBEDTLS_MPI_MAX_BITS. - */ -int mbedtls_mpi_gen_prime( mbedtls_mpi *X, size_t nbits, int flags, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief Checkup routine - * - * \return 0 if successful, or 1 if the test failed - */ -int mbedtls_mpi_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* bignum.h */ diff --git a/windows/mbedtls/build_info.h b/windows/mbedtls/build_info.h deleted file mode 100644 index 23f85ba0..00000000 --- a/windows/mbedtls/build_info.h +++ /dev/null @@ -1,83 +0,0 @@ -/** - * \file build_info.h - * - * \brief Build-time configuration info - * - * Include this file if you need to depend on the - * configuration options defined in mbedtls_config.h or MBEDTLS_CONFIG_FILE - */ - /* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_BUILD_INFO_H -#define MBEDTLS_BUILD_INFO_H - -/* - * This set of compile-time defines can be used to determine the version number - * of the Mbed TLS library used. Run-time variables for the same can be found in - * version.h - */ - -/** - * The version number x.y.z is split into three parts. - * Major, Minor, Patchlevel - */ -#define MBEDTLS_VERSION_MAJOR 3 -#define MBEDTLS_VERSION_MINOR 0 -#define MBEDTLS_VERSION_PATCH 0 - -/** - * The single version number has the following structure: - * MMNNPP00 - * Major version | Minor version | Patch version - */ -#define MBEDTLS_VERSION_NUMBER 0x03000000 -#define MBEDTLS_VERSION_STRING "3.0.0" -#define MBEDTLS_VERSION_STRING_FULL "mbed TLS 3.0.0" - -#if defined(_MSC_VER) && !defined(_CRT_SECURE_NO_DEPRECATE) -#define _CRT_SECURE_NO_DEPRECATE 1 -#endif - -#if !defined(MBEDTLS_CONFIG_FILE) -#include "mbedtls/mbedtls_config.h" -#else -#include MBEDTLS_CONFIG_FILE -#endif - -#if defined(MBEDTLS_CONFIG_VERSION) && ( \ - MBEDTLS_CONFIG_VERSION < 0x03000000 || \ - MBEDTLS_CONFIG_VERSION > MBEDTLS_VERSION_NUMBER ) -#error "Invalid config version, defined value of MBEDTLS_CONFIG_VERSION is unsupported" -#endif - -/* Target and application specific configurations - * - * Allow user to override any previous default. - * - */ -#if defined(MBEDTLS_USER_CONFIG_FILE) -#include MBEDTLS_USER_CONFIG_FILE -#endif - -#if defined(MBEDTLS_PSA_CRYPTO_CONFIG) -#include "mbedtls/config_psa.h" -#endif - -#include "mbedtls/check_config.h" - -#endif /* MBEDTLS_BUILD_INFO_H */ diff --git a/windows/mbedtls/camellia.h b/windows/mbedtls/camellia.h deleted file mode 100644 index 6a09da88..00000000 --- a/windows/mbedtls/camellia.h +++ /dev/null @@ -1,314 +0,0 @@ -/** - * \file camellia.h - * - * \brief Camellia block cipher - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_CAMELLIA_H -#define MBEDTLS_CAMELLIA_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -#include "mbedtls/platform_util.h" - -#define MBEDTLS_CAMELLIA_ENCRYPT 1 -#define MBEDTLS_CAMELLIA_DECRYPT 0 - -#define MBEDTLS_ERR_CAMELLIA_BAD_INPUT_DATA -0x0024 /**< Bad input data. */ - -#define MBEDTLS_ERR_CAMELLIA_INVALID_INPUT_LENGTH -0x0026 /**< Invalid data input length. */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_CAMELLIA_ALT) -// Regular implementation -// - -/** - * \brief CAMELLIA context structure - */ -typedef struct mbedtls_camellia_context -{ - int MBEDTLS_PRIVATE(nr); /*!< number of rounds */ - uint32_t MBEDTLS_PRIVATE(rk)[68]; /*!< CAMELLIA round keys */ -} -mbedtls_camellia_context; - -#else /* MBEDTLS_CAMELLIA_ALT */ -#include "camellia_alt.h" -#endif /* MBEDTLS_CAMELLIA_ALT */ - -/** - * \brief Initialize a CAMELLIA context. - * - * \param ctx The CAMELLIA context to be initialized. - * This must not be \c NULL. - */ -void mbedtls_camellia_init( mbedtls_camellia_context *ctx ); - -/** - * \brief Clear a CAMELLIA context. - * - * \param ctx The CAMELLIA context to be cleared. This may be \c NULL, - * in which case this function returns immediately. If it is not - * \c NULL, it must be initialized. - */ -void mbedtls_camellia_free( mbedtls_camellia_context *ctx ); - -/** - * \brief Perform a CAMELLIA key schedule operation for encryption. - * - * \param ctx The CAMELLIA context to use. This must be initialized. - * \param key The encryption key to use. This must be a readable buffer - * of size \p keybits Bits. - * \param keybits The length of \p key in Bits. This must be either \c 128, - * \c 192 or \c 256. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_camellia_setkey_enc( mbedtls_camellia_context *ctx, - const unsigned char *key, - unsigned int keybits ); - -/** - * \brief Perform a CAMELLIA key schedule operation for decryption. - * - * \param ctx The CAMELLIA context to use. This must be initialized. - * \param key The decryption key. This must be a readable buffer - * of size \p keybits Bits. - * \param keybits The length of \p key in Bits. This must be either \c 128, - * \c 192 or \c 256. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_camellia_setkey_dec( mbedtls_camellia_context *ctx, - const unsigned char *key, - unsigned int keybits ); - -/** - * \brief Perform a CAMELLIA-ECB block encryption/decryption operation. - * - * \param ctx The CAMELLIA context to use. This must be initialized - * and bound to a key. - * \param mode The mode of operation. This must be either - * #MBEDTLS_CAMELLIA_ENCRYPT or #MBEDTLS_CAMELLIA_DECRYPT. - * \param input The input block. This must be a readable buffer - * of size \c 16 Bytes. - * \param output The output block. This must be a writable buffer - * of size \c 16 Bytes. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_camellia_crypt_ecb( mbedtls_camellia_context *ctx, - int mode, - const unsigned char input[16], - unsigned char output[16] ); - -#if defined(MBEDTLS_CIPHER_MODE_CBC) -/** - * \brief Perform a CAMELLIA-CBC buffer encryption/decryption operation. - * - * \note Upon exit, the content of the IV is updated so that you can - * call the function same function again on the following - * block(s) of data and get the same result as if it was - * encrypted in one call. This allows a "streaming" usage. - * If on the other hand you need to retain the contents of the - * IV, you should either save it manually or use the cipher - * module instead. - * - * \param ctx The CAMELLIA context to use. This must be initialized - * and bound to a key. - * \param mode The mode of operation. This must be either - * #MBEDTLS_CAMELLIA_ENCRYPT or #MBEDTLS_CAMELLIA_DECRYPT. - * \param length The length in Bytes of the input data \p input. - * This must be a multiple of \c 16 Bytes. - * \param iv The initialization vector. This must be a read/write buffer - * of length \c 16 Bytes. It is updated to allow streaming - * use as explained above. - * \param input The buffer holding the input data. This must point to a - * readable buffer of length \p length Bytes. - * \param output The buffer holding the output data. This must point to a - * writable buffer of length \p length Bytes. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_camellia_crypt_cbc( mbedtls_camellia_context *ctx, - int mode, - size_t length, - unsigned char iv[16], - const unsigned char *input, - unsigned char *output ); -#endif /* MBEDTLS_CIPHER_MODE_CBC */ - -#if defined(MBEDTLS_CIPHER_MODE_CFB) -/** - * \brief Perform a CAMELLIA-CFB128 buffer encryption/decryption - * operation. - * - * \note Due to the nature of CFB mode, you should use the same - * key for both encryption and decryption. In particular, calls - * to this function should be preceded by a key-schedule via - * mbedtls_camellia_setkey_enc() regardless of whether \p mode - * is #MBEDTLS_CAMELLIA_ENCRYPT or #MBEDTLS_CAMELLIA_DECRYPT. - * - * \note Upon exit, the content of the IV is updated so that you can - * call the function same function again on the following - * block(s) of data and get the same result as if it was - * encrypted in one call. This allows a "streaming" usage. - * If on the other hand you need to retain the contents of the - * IV, you should either save it manually or use the cipher - * module instead. - * - * \param ctx The CAMELLIA context to use. This must be initialized - * and bound to a key. - * \param mode The mode of operation. This must be either - * #MBEDTLS_CAMELLIA_ENCRYPT or #MBEDTLS_CAMELLIA_DECRYPT. - * \param length The length of the input data \p input. Any value is allowed. - * \param iv_off The current offset in the IV. This must be smaller - * than \c 16 Bytes. It is updated after this call to allow - * the aforementioned streaming usage. - * \param iv The initialization vector. This must be a read/write buffer - * of length \c 16 Bytes. It is updated after this call to - * allow the aforementioned streaming usage. - * \param input The buffer holding the input data. This must be a readable - * buffer of size \p length Bytes. - * \param output The buffer to hold the output data. This must be a writable - * buffer of length \p length Bytes. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_camellia_crypt_cfb128( mbedtls_camellia_context *ctx, - int mode, - size_t length, - size_t *iv_off, - unsigned char iv[16], - const unsigned char *input, - unsigned char *output ); -#endif /* MBEDTLS_CIPHER_MODE_CFB */ - -#if defined(MBEDTLS_CIPHER_MODE_CTR) -/** - * \brief Perform a CAMELLIA-CTR buffer encryption/decryption operation. - * - * *note Due to the nature of CTR mode, you should use the same - * key for both encryption and decryption. In particular, calls - * to this function should be preceded by a key-schedule via - * mbedtls_camellia_setkey_enc() regardless of whether \p mode - * is #MBEDTLS_CAMELLIA_ENCRYPT or #MBEDTLS_CAMELLIA_DECRYPT. - * - * \warning You must never reuse a nonce value with the same key. Doing so - * would void the encryption for the two messages encrypted with - * the same nonce and key. - * - * There are two common strategies for managing nonces with CTR: - * - * 1. You can handle everything as a single message processed over - * successive calls to this function. In that case, you want to - * set \p nonce_counter and \p nc_off to 0 for the first call, and - * then preserve the values of \p nonce_counter, \p nc_off and \p - * stream_block across calls to this function as they will be - * updated by this function. - * - * With this strategy, you must not encrypt more than 2**128 - * blocks of data with the same key. - * - * 2. You can encrypt separate messages by dividing the \p - * nonce_counter buffer in two areas: the first one used for a - * per-message nonce, handled by yourself, and the second one - * updated by this function internally. - * - * For example, you might reserve the first \c 12 Bytes for the - * per-message nonce, and the last \c 4 Bytes for internal use. - * In that case, before calling this function on a new message you - * need to set the first \c 12 Bytes of \p nonce_counter to your - * chosen nonce value, the last four to \c 0, and \p nc_off to \c 0 - * (which will cause \p stream_block to be ignored). That way, you - * can encrypt at most \c 2**96 messages of up to \c 2**32 blocks - * each with the same key. - * - * The per-message nonce (or information sufficient to reconstruct - * it) needs to be communicated with the ciphertext and must be - * unique. The recommended way to ensure uniqueness is to use a - * message counter. An alternative is to generate random nonces, - * but this limits the number of messages that can be securely - * encrypted: for example, with 96-bit random nonces, you should - * not encrypt more than 2**32 messages with the same key. - * - * Note that for both stategies, sizes are measured in blocks and - * that a CAMELLIA block is \c 16 Bytes. - * - * \warning Upon return, \p stream_block contains sensitive data. Its - * content must not be written to insecure storage and should be - * securely discarded as soon as it's no longer needed. - * - * \param ctx The CAMELLIA context to use. This must be initialized - * and bound to a key. - * \param length The length of the input data \p input in Bytes. - * Any value is allowed. - * \param nc_off The offset in the current \p stream_block (for resuming - * within current cipher stream). The offset pointer to - * should be \c 0 at the start of a stream. It is updated - * at the end of this call. - * \param nonce_counter The 128-bit nonce and counter. This must be a read/write - * buffer of length \c 16 Bytes. - * \param stream_block The saved stream-block for resuming. This must be a - * read/write buffer of length \c 16 Bytes. - * \param input The input data stream. This must be a readable buffer of - * size \p length Bytes. - * \param output The output data stream. This must be a writable buffer - * of size \p length Bytes. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_camellia_crypt_ctr( mbedtls_camellia_context *ctx, - size_t length, - size_t *nc_off, - unsigned char nonce_counter[16], - unsigned char stream_block[16], - const unsigned char *input, - unsigned char *output ); -#endif /* MBEDTLS_CIPHER_MODE_CTR */ - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief Checkup routine - * - * \return 0 if successful, or 1 if the test failed - */ -int mbedtls_camellia_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* camellia.h */ diff --git a/windows/mbedtls/ccm.h b/windows/mbedtls/ccm.h deleted file mode 100644 index 6613cee7..00000000 --- a/windows/mbedtls/ccm.h +++ /dev/null @@ -1,508 +0,0 @@ -/** - * \file ccm.h - * - * \brief This file provides an API for the CCM authenticated encryption - * mode for block ciphers. - * - * CCM combines Counter mode encryption with CBC-MAC authentication - * for 128-bit block ciphers. - * - * Input to CCM includes the following elements: - *
  • Payload - data that is both authenticated and encrypted.
  • - *
  • Associated data (Adata) - data that is authenticated but not - * encrypted, For example, a header.
  • - *
  • Nonce - A unique value that is assigned to the payload and the - * associated data.
- * - * Definition of CCM: - * http://csrc.nist.gov/publications/nistpubs/800-38C/SP800-38C_updated-July20_2007.pdf - * RFC 3610 "Counter with CBC-MAC (CCM)" - * - * Related: - * RFC 5116 "An Interface and Algorithms for Authenticated Encryption" - * - * Definition of CCM*: - * IEEE 802.15.4 - IEEE Standard for Local and metropolitan area networks - * Integer representation is fixed most-significant-octet-first order and - * the representation of octets is most-significant-bit-first order. This is - * consistent with RFC 3610. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_CCM_H -#define MBEDTLS_CCM_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/cipher.h" - -#define MBEDTLS_CCM_DECRYPT 0 -#define MBEDTLS_CCM_ENCRYPT 1 -#define MBEDTLS_CCM_STAR_DECRYPT 2 -#define MBEDTLS_CCM_STAR_ENCRYPT 3 - -#define MBEDTLS_ERR_CCM_BAD_INPUT -0x000D /**< Bad input parameters to the function. */ -#define MBEDTLS_ERR_CCM_AUTH_FAILED -0x000F /**< Authenticated decryption failed. */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_CCM_ALT) -// Regular implementation -// - -/** - * \brief The CCM context-type definition. The CCM context is passed - * to the APIs called. - */ -typedef struct mbedtls_ccm_context -{ - mbedtls_cipher_context_t MBEDTLS_PRIVATE(cipher_ctx); /*!< The cipher context used. */ -} -mbedtls_ccm_context; - -#else /* MBEDTLS_CCM_ALT */ -#include "ccm_alt.h" -#endif /* MBEDTLS_CCM_ALT */ - -/** - * \brief This function initializes the specified CCM context, - * to make references valid, and prepare the context - * for mbedtls_ccm_setkey() or mbedtls_ccm_free(). - * - * \param ctx The CCM context to initialize. This must not be \c NULL. - */ -void mbedtls_ccm_init( mbedtls_ccm_context *ctx ); - -/** - * \brief This function initializes the CCM context set in the - * \p ctx parameter and sets the encryption key. - * - * \param ctx The CCM context to initialize. This must be an initialized - * context. - * \param cipher The 128-bit block cipher to use. - * \param key The encryption key. This must not be \c NULL. - * \param keybits The key size in bits. This must be acceptable by the cipher. - * - * \return \c 0 on success. - * \return A CCM or cipher-specific error code on failure. - */ -int mbedtls_ccm_setkey( mbedtls_ccm_context *ctx, - mbedtls_cipher_id_t cipher, - const unsigned char *key, - unsigned int keybits ); - -/** - * \brief This function releases and clears the specified CCM context - * and underlying cipher sub-context. - * - * \param ctx The CCM context to clear. If this is \c NULL, the function - * has no effect. Otherwise, this must be initialized. - */ -void mbedtls_ccm_free( mbedtls_ccm_context *ctx ); - -/** - * \brief This function encrypts a buffer using CCM. - * - * \note The tag is written to a separate buffer. To concatenate - * the \p tag with the \p output, as done in RFC-3610: - * Counter with CBC-MAC (CCM), use - * \p tag = \p output + \p length, and make sure that the - * output buffer is at least \p length + \p tag_len wide. - * - * \param ctx The CCM context to use for encryption. This must be - * initialized and bound to a key. - * \param length The length of the input data in Bytes. - * \param iv The initialization vector (nonce). This must be a readable - * buffer of at least \p iv_len Bytes. - * \param iv_len The length of the nonce in Bytes: 7, 8, 9, 10, 11, 12, - * or 13. The length L of the message length field is - * 15 - \p iv_len. - * \param ad The additional data field. If \p ad_len is greater than - * zero, \p ad must be a readable buffer of at least that - * length. - * \param ad_len The length of additional data in Bytes. - * This must be less than `2^16 - 2^8`. - * \param input The buffer holding the input data. If \p length is greater - * than zero, \p input must be a readable buffer of at least - * that length. - * \param output The buffer holding the output data. If \p length is greater - * than zero, \p output must be a writable buffer of at least - * that length. - * \param tag The buffer holding the authentication field. This must be a - * writable buffer of at least \p tag_len Bytes. - * \param tag_len The length of the authentication field to generate in Bytes: - * 4, 6, 8, 10, 12, 14 or 16. - * - * \return \c 0 on success. - * \return A CCM or cipher-specific error code on failure. - */ -int mbedtls_ccm_encrypt_and_tag( mbedtls_ccm_context *ctx, size_t length, - const unsigned char *iv, size_t iv_len, - const unsigned char *ad, size_t ad_len, - const unsigned char *input, unsigned char *output, - unsigned char *tag, size_t tag_len ); - -/** - * \brief This function encrypts a buffer using CCM*. - * - * \note The tag is written to a separate buffer. To concatenate - * the \p tag with the \p output, as done in RFC-3610: - * Counter with CBC-MAC (CCM), use - * \p tag = \p output + \p length, and make sure that the - * output buffer is at least \p length + \p tag_len wide. - * - * \note When using this function in a variable tag length context, - * the tag length has to be encoded into the \p iv passed to - * this function. - * - * \param ctx The CCM context to use for encryption. This must be - * initialized and bound to a key. - * \param length The length of the input data in Bytes. - * \param iv The initialization vector (nonce). This must be a readable - * buffer of at least \p iv_len Bytes. - * \param iv_len The length of the nonce in Bytes: 7, 8, 9, 10, 11, 12, - * or 13. The length L of the message length field is - * 15 - \p iv_len. - * \param ad The additional data field. This must be a readable buffer of - * at least \p ad_len Bytes. - * \param ad_len The length of additional data in Bytes. - * This must be less than 2^16 - 2^8. - * \param input The buffer holding the input data. If \p length is greater - * than zero, \p input must be a readable buffer of at least - * that length. - * \param output The buffer holding the output data. If \p length is greater - * than zero, \p output must be a writable buffer of at least - * that length. - * \param tag The buffer holding the authentication field. This must be a - * writable buffer of at least \p tag_len Bytes. - * \param tag_len The length of the authentication field to generate in Bytes: - * 0, 4, 6, 8, 10, 12, 14 or 16. - * - * \warning Passing \c 0 as \p tag_len means that the message is no - * longer authenticated. - * - * \return \c 0 on success. - * \return A CCM or cipher-specific error code on failure. - */ -int mbedtls_ccm_star_encrypt_and_tag( mbedtls_ccm_context *ctx, size_t length, - const unsigned char *iv, size_t iv_len, - const unsigned char *ad, size_t ad_len, - const unsigned char *input, unsigned char *output, - unsigned char *tag, size_t tag_len ); - -/** - * \brief This function performs a CCM authenticated decryption of a - * buffer. - * - * \param ctx The CCM context to use for decryption. This must be - * initialized and bound to a key. - * \param length The length of the input data in Bytes. - * \param iv The initialization vector (nonce). This must be a readable - * buffer of at least \p iv_len Bytes. - * \param iv_len The length of the nonce in Bytes: 7, 8, 9, 10, 11, 12, - * or 13. The length L of the message length field is - * 15 - \p iv_len. - * \param ad The additional data field. This must be a readable buffer - * of at least that \p ad_len Bytes.. - * \param ad_len The length of additional data in Bytes. - * This must be less than 2^16 - 2^8. - * \param input The buffer holding the input data. If \p length is greater - * than zero, \p input must be a readable buffer of at least - * that length. - * \param output The buffer holding the output data. If \p length is greater - * than zero, \p output must be a writable buffer of at least - * that length. - * \param tag The buffer holding the authentication field. This must be a - * readable buffer of at least \p tag_len Bytes. - * \param tag_len The length of the authentication field to generate in Bytes: - * 4, 6, 8, 10, 12, 14 or 16. - * - * \return \c 0 on success. This indicates that the message is authentic. - * \return #MBEDTLS_ERR_CCM_AUTH_FAILED if the tag does not match. - * \return A cipher-specific error code on calculation failure. - */ -int mbedtls_ccm_auth_decrypt( mbedtls_ccm_context *ctx, size_t length, - const unsigned char *iv, size_t iv_len, - const unsigned char *ad, size_t ad_len, - const unsigned char *input, unsigned char *output, - const unsigned char *tag, size_t tag_len ); - -/** - * \brief This function performs a CCM* authenticated decryption of a - * buffer. - * - * \note When using this function in a variable tag length context, - * the tag length has to be decoded from \p iv and passed to - * this function as \p tag_len. (\p tag needs to be adjusted - * accordingly.) - * - * \param ctx The CCM context to use for decryption. This must be - * initialized and bound to a key. - * \param length The length of the input data in Bytes. - * \param iv The initialization vector (nonce). This must be a readable - * buffer of at least \p iv_len Bytes. - * \param iv_len The length of the nonce in Bytes: 7, 8, 9, 10, 11, 12, - * or 13. The length L of the message length field is - * 15 - \p iv_len. - * \param ad The additional data field. This must be a readable buffer of - * at least that \p ad_len Bytes. - * \param ad_len The length of additional data in Bytes. - * This must be less than 2^16 - 2^8. - * \param input The buffer holding the input data. If \p length is greater - * than zero, \p input must be a readable buffer of at least - * that length. - * \param output The buffer holding the output data. If \p length is greater - * than zero, \p output must be a writable buffer of at least - * that length. - * \param tag The buffer holding the authentication field. This must be a - * readable buffer of at least \p tag_len Bytes. - * \param tag_len The length of the authentication field in Bytes. - * 0, 4, 6, 8, 10, 12, 14 or 16. - * - * \warning Passing \c 0 as \p tag_len means that the message is nos - * longer authenticated. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CCM_AUTH_FAILED if the tag does not match. - * \return A cipher-specific error code on calculation failure. - */ -int mbedtls_ccm_star_auth_decrypt( mbedtls_ccm_context *ctx, size_t length, - const unsigned char *iv, size_t iv_len, - const unsigned char *ad, size_t ad_len, - const unsigned char *input, unsigned char *output, - const unsigned char *tag, size_t tag_len ); - -/** - * \brief This function starts a CCM encryption or decryption - * operation. - * - * This function and mbedtls_ccm_set_lengths() must be called - * before calling mbedtls_ccm_update_ad() or - * mbedtls_ccm_update(). This function can be called before - * or after mbedtls_ccm_set_lengths(). - * - * \note This function is not implemented in Mbed TLS yet. - * - * \param ctx The CCM context. This must be initialized. - * \param mode The operation to perform: #MBEDTLS_CCM_ENCRYPT or - * #MBEDTLS_CCM_DECRYPT or #MBEDTLS_CCM_STAR_ENCRYPT or - * #MBEDTLS_CCM_STAR_DECRYPT. - * \param iv The initialization vector. This must be a readable buffer - * of at least \p iv_len Bytes. - * \param iv_len The length of the nonce in Bytes: 7, 8, 9, 10, 11, 12, - * or 13. The length L of the message length field is - * 15 - \p iv_len. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CCM_BAD_INPUT on failure: - * \p ctx is in an invalid state, - * \p mode is invalid, - * \p iv_len is invalid (lower than \c 7 or greater than - * \c 13). - */ -int mbedtls_ccm_starts( mbedtls_ccm_context *ctx, - int mode, - const unsigned char *iv, - size_t iv_len ); - -/** - * \brief This function declares the lengths of the message - * and additional data for a CCM encryption or decryption - * operation. - * - * This function and mbedtls_ccm_starts() must be called - * before calling mbedtls_ccm_update_ad() or - * mbedtls_ccm_update(). This function can be called before - * or after mbedtls_ccm_starts(). - * - * \note This function is not implemented in Mbed TLS yet. - * - * \param ctx The CCM context. This must be initialized. - * \param total_ad_len The total length of additional data in bytes. - * This must be less than `2^16 - 2^8`. - * \param plaintext_len The length in bytes of the plaintext to encrypt or - * result of the decryption (thus not encompassing the - * additional data that are not encrypted). - * \param tag_len The length of the tag to generate in Bytes: - * 4, 6, 8, 10, 12, 14 or 16. - * For CCM*, zero is also valid. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CCM_BAD_INPUT on failure: - * \p ctx is in an invalid state, - * \p total_ad_len is greater than \c 0xFF00. - */ -int mbedtls_ccm_set_lengths( mbedtls_ccm_context *ctx, - size_t total_ad_len, - size_t plaintext_len, - size_t tag_len ); - -/** - * \brief This function feeds an input buffer as associated data - * (authenticated but not encrypted data) in a CCM - * encryption or decryption operation. - * - * You may call this function zero, one or more times - * to pass successive parts of the additional data. The - * lengths \p ad_len of the data parts should eventually add - * up exactly to the total length of additional data - * \c total_ad_len passed to mbedtls_ccm_set_lengths(). You - * may not call this function after calling - * mbedtls_ccm_update(). - * - * \note This function is not implemented in Mbed TLS yet. - * - * \param ctx The CCM context. This must have been started with - * mbedtls_ccm_starts(), the lengths of the message and - * additional data must have been declared with - * mbedtls_ccm_set_lengths() and this must not have yet - * received any input with mbedtls_ccm_update(). - * \param ad The buffer holding the additional data, or \c NULL - * if \p ad_len is \c 0. - * \param ad_len The length of the additional data. If \c 0, - * \p ad may be \c NULL. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CCM_BAD_INPUT on failure: - * \p ctx is in an invalid state, - * total input length too long. - */ -int mbedtls_ccm_update_ad( mbedtls_ccm_context *ctx, - const unsigned char *ad, - size_t ad_len ); - -/** - * \brief This function feeds an input buffer into an ongoing CCM - * encryption or decryption operation. - * - * You may call this function zero, one or more times - * to pass successive parts of the input: the plaintext to - * encrypt, or the ciphertext (not including the tag) to - * decrypt. After the last part of the input, call - * mbedtls_ccm_finish(). The lengths \p input_len of the - * data parts should eventually add up exactly to the - * plaintext length \c plaintext_len passed to - * mbedtls_ccm_set_lengths(). - * - * This function may produce output in one of the following - * ways: - * - Immediate output: the output length is always equal - * to the input length. - * - Buffered output: except for the last part of input data, - * the output consists of a whole number of 16-byte blocks. - * If the total input length so far (not including - * associated data) is 16 \* *B* + *A* with *A* < 16 then - * the total output length is 16 \* *B*. - * For the last part of input data, the output length is - * equal to the input length plus the number of bytes (*A*) - * buffered in the previous call to the function (if any). - * The function uses the plaintext length - * \c plaintext_len passed to mbedtls_ccm_set_lengths() - * to detect the last part of input data. - * - * In particular: - * - It is always correct to call this function with - * \p output_size >= \p input_len + 15. - * - If \p input_len is a multiple of 16 for all the calls - * to this function during an operation (not necessary for - * the last one) then it is correct to use \p output_size - * =\p input_len. - * - * \note This function is not implemented in Mbed TLS yet. - * - * \param ctx The CCM context. This must have been started with - * mbedtls_ccm_starts() and the lengths of the message and - * additional data must have been declared with - * mbedtls_ccm_set_lengths(). - * \param input The buffer holding the input data. If \p input_len - * is greater than zero, this must be a readable buffer - * of at least \p input_len bytes. - * \param input_len The length of the input data in bytes. - * \param output The buffer for the output data. If \p output_size - * is greater than zero, this must be a writable buffer of - * at least \p output_size bytes. - * \param output_size The size of the output buffer in bytes. - * See the function description regarding the output size. - * \param output_len On success, \p *output_len contains the actual - * length of the output written in \p output. - * On failure, the content of \p *output_len is - * unspecified. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CCM_BAD_INPUT on failure: - * \p ctx is in an invalid state, - * total input length too long, - * or \p output_size too small. - */ -int mbedtls_ccm_update( mbedtls_ccm_context *ctx, - const unsigned char *input, size_t input_len, - unsigned char *output, size_t output_size, - size_t *output_len ); - -/** - * \brief This function finishes the CCM operation and generates - * the authentication tag. - * - * It wraps up the CCM stream, and generates the - * tag. The tag can have a maximum length of 16 Bytes. - * - * \note This function is not implemented in Mbed TLS yet. - * - * \param ctx The CCM context. This must have been started with - * mbedtls_ccm_starts() and the lengths of the message and - * additional data must have been declared with - * mbedtls_ccm_set_lengths(). - * \param tag The buffer for holding the tag. If \p tag_len is greater - * than zero, this must be a writable buffer of at least \p - * tag_len Bytes. - * \param tag_len The length of the tag. Must match the tag length passed to - * mbedtls_ccm_set_lengths() function. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CCM_BAD_INPUT on failure: - * \p ctx is in an invalid state, - * invalid value of \p tag_len, - * the total amount of additional data passed to - * mbedtls_ccm_update_ad() was lower than the total length of - * additional data \c total_ad_len passed to - * mbedtls_ccm_set_lengths(), - * the total amount of input data passed to - * mbedtls_ccm_update() was lower than the plaintext length - * \c plaintext_len passed to mbedtls_ccm_set_lengths(). - */ -int mbedtls_ccm_finish( mbedtls_ccm_context *ctx, - unsigned char *tag, size_t tag_len ); - -#if defined(MBEDTLS_SELF_TEST) && defined(MBEDTLS_AES_C) -/** - * \brief The CCM checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_ccm_self_test( int verbose ); -#endif /* MBEDTLS_SELF_TEST && MBEDTLS_AES_C */ - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_CCM_H */ diff --git a/windows/mbedtls/chacha20.h b/windows/mbedtls/chacha20.h deleted file mode 100644 index 25e1a263..00000000 --- a/windows/mbedtls/chacha20.h +++ /dev/null @@ -1,214 +0,0 @@ -/** - * \file chacha20.h - * - * \brief This file contains ChaCha20 definitions and functions. - * - * ChaCha20 is a stream cipher that can encrypt and decrypt - * information. ChaCha was created by Daniel Bernstein as a variant of - * its Salsa cipher https://cr.yp.to/chacha/chacha-20080128.pdf - * ChaCha20 is the variant with 20 rounds, that was also standardized - * in RFC 7539. - * - * \author Daniel King - */ - -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_CHACHA20_H -#define MBEDTLS_CHACHA20_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -#define MBEDTLS_ERR_CHACHA20_BAD_INPUT_DATA -0x0051 /**< Invalid input parameter(s). */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_CHACHA20_ALT) - -typedef struct mbedtls_chacha20_context -{ - uint32_t MBEDTLS_PRIVATE(state)[16]; /*! The state (before round operations). */ - uint8_t MBEDTLS_PRIVATE(keystream8)[64]; /*! Leftover keystream bytes. */ - size_t MBEDTLS_PRIVATE(keystream_bytes_used); /*! Number of keystream bytes already used. */ -} -mbedtls_chacha20_context; - -#else /* MBEDTLS_CHACHA20_ALT */ -#include "chacha20_alt.h" -#endif /* MBEDTLS_CHACHA20_ALT */ - -/** - * \brief This function initializes the specified ChaCha20 context. - * - * It must be the first API called before using - * the context. - * - * It is usually followed by calls to - * \c mbedtls_chacha20_setkey() and - * \c mbedtls_chacha20_starts(), then one or more calls to - * to \c mbedtls_chacha20_update(), and finally to - * \c mbedtls_chacha20_free(). - * - * \param ctx The ChaCha20 context to initialize. - * This must not be \c NULL. - */ -void mbedtls_chacha20_init( mbedtls_chacha20_context *ctx ); - -/** - * \brief This function releases and clears the specified - * ChaCha20 context. - * - * \param ctx The ChaCha20 context to clear. This may be \c NULL, - * in which case this function is a no-op. If it is not - * \c NULL, it must point to an initialized context. - * - */ -void mbedtls_chacha20_free( mbedtls_chacha20_context *ctx ); - -/** - * \brief This function sets the encryption/decryption key. - * - * \note After using this function, you must also call - * \c mbedtls_chacha20_starts() to set a nonce before you - * start encrypting/decrypting data with - * \c mbedtls_chacha_update(). - * - * \param ctx The ChaCha20 context to which the key should be bound. - * It must be initialized. - * \param key The encryption/decryption key. This must be \c 32 Bytes - * in length. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CHACHA20_BAD_INPUT_DATA if ctx or key is NULL. - */ -int mbedtls_chacha20_setkey( mbedtls_chacha20_context *ctx, - const unsigned char key[32] ); - -/** - * \brief This function sets the nonce and initial counter value. - * - * \note A ChaCha20 context can be re-used with the same key by - * calling this function to change the nonce. - * - * \warning You must never use the same nonce twice with the same key. - * This would void any confidentiality guarantees for the - * messages encrypted with the same nonce and key. - * - * \param ctx The ChaCha20 context to which the nonce should be bound. - * It must be initialized and bound to a key. - * \param nonce The nonce. This must be \c 12 Bytes in size. - * \param counter The initial counter value. This is usually \c 0. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CHACHA20_BAD_INPUT_DATA if ctx or nonce is - * NULL. - */ -int mbedtls_chacha20_starts( mbedtls_chacha20_context* ctx, - const unsigned char nonce[12], - uint32_t counter ); - -/** - * \brief This function encrypts or decrypts data. - * - * Since ChaCha20 is a stream cipher, the same operation is - * used for encrypting and decrypting data. - * - * \note The \p input and \p output pointers must either be equal or - * point to non-overlapping buffers. - * - * \note \c mbedtls_chacha20_setkey() and - * \c mbedtls_chacha20_starts() must be called at least once - * to setup the context before this function can be called. - * - * \note This function can be called multiple times in a row in - * order to encrypt of decrypt data piecewise with the same - * key and nonce. - * - * \param ctx The ChaCha20 context to use for encryption or decryption. - * It must be initialized and bound to a key and nonce. - * \param size The length of the input data in Bytes. - * \param input The buffer holding the input data. - * This pointer can be \c NULL if `size == 0`. - * \param output The buffer holding the output data. - * This must be able to hold \p size Bytes. - * This pointer can be \c NULL if `size == 0`. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_chacha20_update( mbedtls_chacha20_context *ctx, - size_t size, - const unsigned char *input, - unsigned char *output ); - -/** - * \brief This function encrypts or decrypts data with ChaCha20 and - * the given key and nonce. - * - * Since ChaCha20 is a stream cipher, the same operation is - * used for encrypting and decrypting data. - * - * \warning You must never use the same (key, nonce) pair more than - * once. This would void any confidentiality guarantees for - * the messages encrypted with the same nonce and key. - * - * \note The \p input and \p output pointers must either be equal or - * point to non-overlapping buffers. - * - * \param key The encryption/decryption key. - * This must be \c 32 Bytes in length. - * \param nonce The nonce. This must be \c 12 Bytes in size. - * \param counter The initial counter value. This is usually \c 0. - * \param size The length of the input data in Bytes. - * \param input The buffer holding the input data. - * This pointer can be \c NULL if `size == 0`. - * \param output The buffer holding the output data. - * This must be able to hold \p size Bytes. - * This pointer can be \c NULL if `size == 0`. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_chacha20_crypt( const unsigned char key[32], - const unsigned char nonce[12], - uint32_t counter, - size_t size, - const unsigned char* input, - unsigned char* output ); - -#if defined(MBEDTLS_SELF_TEST) -/** - * \brief The ChaCha20 checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_chacha20_self_test( int verbose ); -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_CHACHA20_H */ diff --git a/windows/mbedtls/chachapoly.h b/windows/mbedtls/chachapoly.h deleted file mode 100644 index 3f6e4a33..00000000 --- a/windows/mbedtls/chachapoly.h +++ /dev/null @@ -1,354 +0,0 @@ -/** - * \file chachapoly.h - * - * \brief This file contains the AEAD-ChaCha20-Poly1305 definitions and - * functions. - * - * ChaCha20-Poly1305 is an algorithm for Authenticated Encryption - * with Associated Data (AEAD) that can be used to encrypt and - * authenticate data. It is based on ChaCha20 and Poly1305 by Daniel - * Bernstein and was standardized in RFC 7539. - * - * \author Daniel King - */ - -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_CHACHAPOLY_H -#define MBEDTLS_CHACHAPOLY_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -/* for shared error codes */ -#include "mbedtls/poly1305.h" - -#define MBEDTLS_ERR_CHACHAPOLY_BAD_STATE -0x0054 /**< The requested operation is not permitted in the current state. */ -#define MBEDTLS_ERR_CHACHAPOLY_AUTH_FAILED -0x0056 /**< Authenticated decryption failed: data was not authentic. */ - -#ifdef __cplusplus -extern "C" { -#endif - -typedef enum -{ - MBEDTLS_CHACHAPOLY_ENCRYPT, /**< The mode value for performing encryption. */ - MBEDTLS_CHACHAPOLY_DECRYPT /**< The mode value for performing decryption. */ -} -mbedtls_chachapoly_mode_t; - -#if !defined(MBEDTLS_CHACHAPOLY_ALT) - -#include "mbedtls/chacha20.h" - -typedef struct mbedtls_chachapoly_context -{ - mbedtls_chacha20_context MBEDTLS_PRIVATE(chacha20_ctx); /**< The ChaCha20 context. */ - mbedtls_poly1305_context MBEDTLS_PRIVATE(poly1305_ctx); /**< The Poly1305 context. */ - uint64_t MBEDTLS_PRIVATE(aad_len); /**< The length (bytes) of the Additional Authenticated Data. */ - uint64_t MBEDTLS_PRIVATE(ciphertext_len); /**< The length (bytes) of the ciphertext. */ - int MBEDTLS_PRIVATE(state); /**< The current state of the context. */ - mbedtls_chachapoly_mode_t MBEDTLS_PRIVATE(mode); /**< Cipher mode (encrypt or decrypt). */ -} -mbedtls_chachapoly_context; - -#else /* !MBEDTLS_CHACHAPOLY_ALT */ -#include "chachapoly_alt.h" -#endif /* !MBEDTLS_CHACHAPOLY_ALT */ - -/** - * \brief This function initializes the specified ChaCha20-Poly1305 context. - * - * It must be the first API called before using - * the context. It must be followed by a call to - * \c mbedtls_chachapoly_setkey() before any operation can be - * done, and to \c mbedtls_chachapoly_free() once all - * operations with that context have been finished. - * - * In order to encrypt or decrypt full messages at once, for - * each message you should make a single call to - * \c mbedtls_chachapoly_crypt_and_tag() or - * \c mbedtls_chachapoly_auth_decrypt(). - * - * In order to encrypt messages piecewise, for each - * message you should make a call to - * \c mbedtls_chachapoly_starts(), then 0 or more calls to - * \c mbedtls_chachapoly_update_aad(), then 0 or more calls to - * \c mbedtls_chachapoly_update(), then one call to - * \c mbedtls_chachapoly_finish(). - * - * \warning Decryption with the piecewise API is discouraged! Always - * use \c mbedtls_chachapoly_auth_decrypt() when possible! - * - * If however this is not possible because the data is too - * large to fit in memory, you need to: - * - * - call \c mbedtls_chachapoly_starts() and (if needed) - * \c mbedtls_chachapoly_update_aad() as above, - * - call \c mbedtls_chachapoly_update() multiple times and - * ensure its output (the plaintext) is NOT used in any other - * way than placing it in temporary storage at this point, - * - call \c mbedtls_chachapoly_finish() to compute the - * authentication tag and compared it in constant time to the - * tag received with the ciphertext. - * - * If the tags are not equal, you must immediately discard - * all previous outputs of \c mbedtls_chachapoly_update(), - * otherwise you can now safely use the plaintext. - * - * \param ctx The ChachaPoly context to initialize. Must not be \c NULL. - */ -void mbedtls_chachapoly_init( mbedtls_chachapoly_context *ctx ); - -/** - * \brief This function releases and clears the specified - * ChaCha20-Poly1305 context. - * - * \param ctx The ChachaPoly context to clear. This may be \c NULL, in which - * case this function is a no-op. - */ -void mbedtls_chachapoly_free( mbedtls_chachapoly_context *ctx ); - -/** - * \brief This function sets the ChaCha20-Poly1305 - * symmetric encryption key. - * - * \param ctx The ChaCha20-Poly1305 context to which the key should be - * bound. This must be initialized. - * \param key The \c 256 Bit (\c 32 Bytes) key. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_chachapoly_setkey( mbedtls_chachapoly_context *ctx, - const unsigned char key[32] ); - -/** - * \brief This function starts a ChaCha20-Poly1305 encryption or - * decryption operation. - * - * \warning You must never use the same nonce twice with the same key. - * This would void any confidentiality and authenticity - * guarantees for the messages encrypted with the same nonce - * and key. - * - * \note If the context is being used for AAD only (no data to - * encrypt or decrypt) then \p mode can be set to any value. - * - * \warning Decryption with the piecewise API is discouraged, see the - * warning on \c mbedtls_chachapoly_init(). - * - * \param ctx The ChaCha20-Poly1305 context. This must be initialized - * and bound to a key. - * \param nonce The nonce/IV to use for the message. - * This must be a redable buffer of length \c 12 Bytes. - * \param mode The operation to perform: #MBEDTLS_CHACHAPOLY_ENCRYPT or - * #MBEDTLS_CHACHAPOLY_DECRYPT (discouraged, see warning). - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_chachapoly_starts( mbedtls_chachapoly_context *ctx, - const unsigned char nonce[12], - mbedtls_chachapoly_mode_t mode ); - -/** - * \brief This function feeds additional data to be authenticated - * into an ongoing ChaCha20-Poly1305 operation. - * - * The Additional Authenticated Data (AAD), also called - * Associated Data (AD) is only authenticated but not - * encrypted nor included in the encrypted output. It is - * usually transmitted separately from the ciphertext or - * computed locally by each party. - * - * \note This function is called before data is encrypted/decrypted. - * I.e. call this function to process the AAD before calling - * \c mbedtls_chachapoly_update(). - * - * You may call this function multiple times to process - * an arbitrary amount of AAD. It is permitted to call - * this function 0 times, if no AAD is used. - * - * This function cannot be called any more if data has - * been processed by \c mbedtls_chachapoly_update(), - * or if the context has been finished. - * - * \warning Decryption with the piecewise API is discouraged, see the - * warning on \c mbedtls_chachapoly_init(). - * - * \param ctx The ChaCha20-Poly1305 context. This must be initialized - * and bound to a key. - * \param aad_len The length in Bytes of the AAD. The length has no - * restrictions. - * \param aad Buffer containing the AAD. - * This pointer can be \c NULL if `aad_len == 0`. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_POLY1305_BAD_INPUT_DATA - * if \p ctx or \p aad are NULL. - * \return #MBEDTLS_ERR_CHACHAPOLY_BAD_STATE - * if the operations has not been started or has been - * finished, or if the AAD has been finished. - */ -int mbedtls_chachapoly_update_aad( mbedtls_chachapoly_context *ctx, - const unsigned char *aad, - size_t aad_len ); - -/** - * \brief Thus function feeds data to be encrypted or decrypted - * into an on-going ChaCha20-Poly1305 - * operation. - * - * The direction (encryption or decryption) depends on the - * mode that was given when calling - * \c mbedtls_chachapoly_starts(). - * - * You may call this function multiple times to process - * an arbitrary amount of data. It is permitted to call - * this function 0 times, if no data is to be encrypted - * or decrypted. - * - * \warning Decryption with the piecewise API is discouraged, see the - * warning on \c mbedtls_chachapoly_init(). - * - * \param ctx The ChaCha20-Poly1305 context to use. This must be initialized. - * \param len The length (in bytes) of the data to encrypt or decrypt. - * \param input The buffer containing the data to encrypt or decrypt. - * This pointer can be \c NULL if `len == 0`. - * \param output The buffer to where the encrypted or decrypted data is - * written. This must be able to hold \p len bytes. - * This pointer can be \c NULL if `len == 0`. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CHACHAPOLY_BAD_STATE - * if the operation has not been started or has been - * finished. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_chachapoly_update( mbedtls_chachapoly_context *ctx, - size_t len, - const unsigned char *input, - unsigned char *output ); - -/** - * \brief This function finished the ChaCha20-Poly1305 operation and - * generates the MAC (authentication tag). - * - * \param ctx The ChaCha20-Poly1305 context to use. This must be initialized. - * \param mac The buffer to where the 128-bit (16 bytes) MAC is written. - * - * \warning Decryption with the piecewise API is discouraged, see the - * warning on \c mbedtls_chachapoly_init(). - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CHACHAPOLY_BAD_STATE - * if the operation has not been started or has been - * finished. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_chachapoly_finish( mbedtls_chachapoly_context *ctx, - unsigned char mac[16] ); - -/** - * \brief This function performs a complete ChaCha20-Poly1305 - * authenticated encryption with the previously-set key. - * - * \note Before using this function, you must set the key with - * \c mbedtls_chachapoly_setkey(). - * - * \warning You must never use the same nonce twice with the same key. - * This would void any confidentiality and authenticity - * guarantees for the messages encrypted with the same nonce - * and key. - * - * \param ctx The ChaCha20-Poly1305 context to use (holds the key). - * This must be initialized. - * \param length The length (in bytes) of the data to encrypt or decrypt. - * \param nonce The 96-bit (12 bytes) nonce/IV to use. - * \param aad The buffer containing the additional authenticated - * data (AAD). This pointer can be \c NULL if `aad_len == 0`. - * \param aad_len The length (in bytes) of the AAD data to process. - * \param input The buffer containing the data to encrypt or decrypt. - * This pointer can be \c NULL if `ilen == 0`. - * \param output The buffer to where the encrypted or decrypted data - * is written. This pointer can be \c NULL if `ilen == 0`. - * \param tag The buffer to where the computed 128-bit (16 bytes) MAC - * is written. This must not be \c NULL. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_chachapoly_encrypt_and_tag( mbedtls_chachapoly_context *ctx, - size_t length, - const unsigned char nonce[12], - const unsigned char *aad, - size_t aad_len, - const unsigned char *input, - unsigned char *output, - unsigned char tag[16] ); - -/** - * \brief This function performs a complete ChaCha20-Poly1305 - * authenticated decryption with the previously-set key. - * - * \note Before using this function, you must set the key with - * \c mbedtls_chachapoly_setkey(). - * - * \param ctx The ChaCha20-Poly1305 context to use (holds the key). - * \param length The length (in Bytes) of the data to decrypt. - * \param nonce The \c 96 Bit (\c 12 bytes) nonce/IV to use. - * \param aad The buffer containing the additional authenticated data (AAD). - * This pointer can be \c NULL if `aad_len == 0`. - * \param aad_len The length (in bytes) of the AAD data to process. - * \param tag The buffer holding the authentication tag. - * This must be a readable buffer of length \c 16 Bytes. - * \param input The buffer containing the data to decrypt. - * This pointer can be \c NULL if `ilen == 0`. - * \param output The buffer to where the decrypted data is written. - * This pointer can be \c NULL if `ilen == 0`. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CHACHAPOLY_AUTH_FAILED - * if the data was not authentic. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_chachapoly_auth_decrypt( mbedtls_chachapoly_context *ctx, - size_t length, - const unsigned char nonce[12], - const unsigned char *aad, - size_t aad_len, - const unsigned char tag[16], - const unsigned char *input, - unsigned char *output ); - -#if defined(MBEDTLS_SELF_TEST) -/** - * \brief The ChaCha20-Poly1305 checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_chachapoly_self_test( int verbose ); -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_CHACHAPOLY_H */ diff --git a/windows/mbedtls/check_config.h b/windows/mbedtls/check_config.h deleted file mode 100644 index e38892d8..00000000 --- a/windows/mbedtls/check_config.h +++ /dev/null @@ -1,838 +0,0 @@ -/** - * \file check_config.h - * - * \brief Consistency checks for configuration options - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_CHECK_CONFIG_H -#define MBEDTLS_CHECK_CONFIG_H - -/* - * We assume CHAR_BIT is 8 in many places. In practice, this is true on our - * target platforms, so not an issue, but let's just be extra sure. - */ -#include -#if CHAR_BIT != 8 -#error "mbed TLS requires a platform with 8-bit chars" -#endif - -#if defined(_WIN32) -#if !defined(MBEDTLS_PLATFORM_C) -#error "MBEDTLS_PLATFORM_C is required on Windows" -#endif - -/* Fix the config here. Not convenient to put an #ifdef _WIN32 in mbedtls_config.h as - * it would confuse config.py. */ -#if !defined(MBEDTLS_PLATFORM_SNPRINTF_ALT) && \ - !defined(MBEDTLS_PLATFORM_SNPRINTF_MACRO) -#define MBEDTLS_PLATFORM_SNPRINTF_ALT -#endif - -#if !defined(MBEDTLS_PLATFORM_VSNPRINTF_ALT) && \ - !defined(MBEDTLS_PLATFORM_VSNPRINTF_MACRO) -#define MBEDTLS_PLATFORM_VSNPRINTF_ALT -#endif -#endif /* _WIN32 */ - -#if defined(TARGET_LIKE_MBED) && defined(MBEDTLS_NET_C) -#error "The NET module is not available for mbed OS - please use the network functions provided by Mbed OS" -#endif - -#if defined(MBEDTLS_DEPRECATED_WARNING) && \ - !defined(__GNUC__) && !defined(__clang__) -#error "MBEDTLS_DEPRECATED_WARNING only works with GCC and Clang" -#endif - -#if defined(MBEDTLS_HAVE_TIME_DATE) && !defined(MBEDTLS_HAVE_TIME) -#error "MBEDTLS_HAVE_TIME_DATE without MBEDTLS_HAVE_TIME does not make sense" -#endif - -#if defined(MBEDTLS_AESNI_C) && !defined(MBEDTLS_HAVE_ASM) -#error "MBEDTLS_AESNI_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_CTR_DRBG_C) && !defined(MBEDTLS_AES_C) -#error "MBEDTLS_CTR_DRBG_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_DHM_C) && !defined(MBEDTLS_BIGNUM_C) -#error "MBEDTLS_DHM_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_CMAC_C) && \ - !defined(MBEDTLS_AES_C) && !defined(MBEDTLS_DES_C) -#error "MBEDTLS_CMAC_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_NIST_KW_C) && \ - ( !defined(MBEDTLS_AES_C) || !defined(MBEDTLS_CIPHER_C) ) -#error "MBEDTLS_NIST_KW_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECDH_C) && !defined(MBEDTLS_ECP_C) -#error "MBEDTLS_ECDH_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECDSA_C) && \ - ( !defined(MBEDTLS_ECP_C) || \ - !( defined(MBEDTLS_ECP_DP_SECP192R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP224R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP256R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP384R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP521R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP192K1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP224K1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP256K1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_BP256R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_BP384R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_BP512R1_ENABLED) ) || \ - !defined(MBEDTLS_ASN1_PARSE_C) || \ - !defined(MBEDTLS_ASN1_WRITE_C) ) -#error "MBEDTLS_ECDSA_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECJPAKE_C) && \ - ( !defined(MBEDTLS_ECP_C) || !defined(MBEDTLS_MD_C) ) -#error "MBEDTLS_ECJPAKE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECP_RESTARTABLE) && \ - ( defined(MBEDTLS_USE_PSA_CRYPTO) || \ - defined(MBEDTLS_ECDH_COMPUTE_SHARED_ALT) || \ - defined(MBEDTLS_ECDH_GEN_PUBLIC_ALT) || \ - defined(MBEDTLS_ECDSA_SIGN_ALT) || \ - defined(MBEDTLS_ECDSA_VERIFY_ALT) || \ - defined(MBEDTLS_ECDSA_GENKEY_ALT) || \ - defined(MBEDTLS_ECP_INTERNAL_ALT) || \ - defined(MBEDTLS_ECP_ALT) ) -#error "MBEDTLS_ECP_RESTARTABLE defined, but it cannot coexist with an alternative or PSA-based ECP implementation" -#endif - -#if defined(MBEDTLS_ECDSA_DETERMINISTIC) && !defined(MBEDTLS_HMAC_DRBG_C) -#error "MBEDTLS_ECDSA_DETERMINISTIC defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECP_C) && ( !defined(MBEDTLS_BIGNUM_C) || ( \ - !defined(MBEDTLS_ECP_DP_SECP192R1_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_SECP224R1_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_SECP256R1_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_SECP384R1_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_SECP521R1_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_BP256R1_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_BP384R1_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_BP512R1_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_SECP192K1_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_SECP224K1_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_SECP256K1_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_CURVE25519_ENABLED) && \ - !defined(MBEDTLS_ECP_DP_CURVE448_ENABLED) ) ) -#error "MBEDTLS_ECP_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PK_PARSE_C) && !defined(MBEDTLS_ASN1_PARSE_C) -#error "MBEDTLS_PK_PARSE_C defined, but not all prerequesites" -#endif - -#if defined(MBEDTLS_ENTROPY_C) && (!defined(MBEDTLS_SHA512_C) && \ - !defined(MBEDTLS_SHA256_C)) -#error "MBEDTLS_ENTROPY_C defined, but not all prerequisites" -#endif -#if defined(MBEDTLS_ENTROPY_C) && defined(MBEDTLS_SHA512_C) && \ - defined(MBEDTLS_CTR_DRBG_ENTROPY_LEN) && (MBEDTLS_CTR_DRBG_ENTROPY_LEN > 64) -#error "MBEDTLS_CTR_DRBG_ENTROPY_LEN value too high" -#endif -#if defined(MBEDTLS_ENTROPY_C) && \ - ( !defined(MBEDTLS_SHA512_C) || defined(MBEDTLS_ENTROPY_FORCE_SHA256) ) \ - && defined(MBEDTLS_CTR_DRBG_ENTROPY_LEN) && (MBEDTLS_CTR_DRBG_ENTROPY_LEN > 32) -#error "MBEDTLS_CTR_DRBG_ENTROPY_LEN value too high" -#endif -#if defined(MBEDTLS_ENTROPY_C) && \ - defined(MBEDTLS_ENTROPY_FORCE_SHA256) && !defined(MBEDTLS_SHA256_C) -#error "MBEDTLS_ENTROPY_FORCE_SHA256 defined, but not all prerequisites" -#endif - -#if defined(__has_feature) -#if __has_feature(memory_sanitizer) -#define MBEDTLS_HAS_MEMSAN -#endif -#endif -#if defined(MBEDTLS_TEST_CONSTANT_FLOW_MEMSAN) && !defined(MBEDTLS_HAS_MEMSAN) -#error "MBEDTLS_TEST_CONSTANT_FLOW_MEMSAN requires building with MemorySanitizer" -#endif -#undef MBEDTLS_HAS_MEMSAN - -#if defined(MBEDTLS_GCM_C) && ( \ - !defined(MBEDTLS_AES_C) && !defined(MBEDTLS_CAMELLIA_C) && !defined(MBEDTLS_ARIA_C) ) -#error "MBEDTLS_GCM_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECP_RANDOMIZE_JAC_ALT) && !defined(MBEDTLS_ECP_INTERNAL_ALT) -#error "MBEDTLS_ECP_RANDOMIZE_JAC_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECP_ADD_MIXED_ALT) && !defined(MBEDTLS_ECP_INTERNAL_ALT) -#error "MBEDTLS_ECP_ADD_MIXED_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECP_DOUBLE_JAC_ALT) && !defined(MBEDTLS_ECP_INTERNAL_ALT) -#error "MBEDTLS_ECP_DOUBLE_JAC_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECP_NORMALIZE_JAC_MANY_ALT) && !defined(MBEDTLS_ECP_INTERNAL_ALT) -#error "MBEDTLS_ECP_NORMALIZE_JAC_MANY_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECP_NORMALIZE_JAC_ALT) && !defined(MBEDTLS_ECP_INTERNAL_ALT) -#error "MBEDTLS_ECP_NORMALIZE_JAC_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECP_DOUBLE_ADD_MXZ_ALT) && !defined(MBEDTLS_ECP_INTERNAL_ALT) -#error "MBEDTLS_ECP_DOUBLE_ADD_MXZ_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECP_RANDOMIZE_MXZ_ALT) && !defined(MBEDTLS_ECP_INTERNAL_ALT) -#error "MBEDTLS_ECP_RANDOMIZE_MXZ_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECP_NORMALIZE_MXZ_ALT) && !defined(MBEDTLS_ECP_INTERNAL_ALT) -#error "MBEDTLS_ECP_NORMALIZE_MXZ_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ECP_NO_FALLBACK) && !defined(MBEDTLS_ECP_INTERNAL_ALT) -#error "MBEDTLS_ECP_NO_FALLBACK defined, but no alternative implementation enabled" -#endif - -#if defined(MBEDTLS_HKDF_C) && !defined(MBEDTLS_MD_C) -#error "MBEDTLS_HKDF_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_HMAC_DRBG_C) && !defined(MBEDTLS_MD_C) -#error "MBEDTLS_HMAC_DRBG_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED) && \ - ( !defined(MBEDTLS_ECDH_C) || !defined(MBEDTLS_ECDSA_C) || \ - !defined(MBEDTLS_X509_CRT_PARSE_C) ) -#error "MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED) && \ - ( !defined(MBEDTLS_ECDH_C) || !defined(MBEDTLS_RSA_C) || \ - !defined(MBEDTLS_X509_CRT_PARSE_C) ) -#error "MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_DHE_PSK_ENABLED) && !defined(MBEDTLS_DHM_C) -#error "MBEDTLS_KEY_EXCHANGE_DHE_PSK_ENABLED defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED) && \ - !defined(MBEDTLS_ECDH_C) -#error "MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED) && \ - ( !defined(MBEDTLS_DHM_C) || !defined(MBEDTLS_RSA_C) || \ - !defined(MBEDTLS_X509_CRT_PARSE_C) || !defined(MBEDTLS_PKCS1_V15) ) -#error "MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED) && \ - ( !defined(MBEDTLS_ECDH_C) || !defined(MBEDTLS_RSA_C) || \ - !defined(MBEDTLS_X509_CRT_PARSE_C) || !defined(MBEDTLS_PKCS1_V15) ) -#error "MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED) && \ - ( !defined(MBEDTLS_ECDH_C) || !defined(MBEDTLS_ECDSA_C) || \ - !defined(MBEDTLS_X509_CRT_PARSE_C) ) -#error "MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED) && \ - ( !defined(MBEDTLS_RSA_C) || !defined(MBEDTLS_X509_CRT_PARSE_C) || \ - !defined(MBEDTLS_PKCS1_V15) ) -#error "MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_RSA_ENABLED) && \ - ( !defined(MBEDTLS_RSA_C) || !defined(MBEDTLS_X509_CRT_PARSE_C) || \ - !defined(MBEDTLS_PKCS1_V15) ) -#error "MBEDTLS_KEY_EXCHANGE_RSA_ENABLED defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED) && \ - ( !defined(MBEDTLS_ECJPAKE_C) || !defined(MBEDTLS_SHA256_C) || \ - !defined(MBEDTLS_ECP_DP_SECP256R1_ENABLED) ) -#error "MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_WITH_CERT_ENABLED) && \ - !defined(MBEDTLS_SSL_KEEP_PEER_CERTIFICATE) && \ - ( !defined(MBEDTLS_SHA256_C) && \ - !defined(MBEDTLS_SHA512_C) && \ - !defined(MBEDTLS_SHA1_C) ) -#error "!MBEDTLS_SSL_KEEP_PEER_CERTIFICATE requires MBEDTLS_SHA512_C, MBEDTLS_SHA256_C or MBEDTLS_SHA1_C" -#endif - -#if defined(MBEDTLS_MEMORY_BUFFER_ALLOC_C) && \ - ( !defined(MBEDTLS_PLATFORM_C) || !defined(MBEDTLS_PLATFORM_MEMORY) ) -#error "MBEDTLS_MEMORY_BUFFER_ALLOC_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_MEMORY_BACKTRACE) && !defined(MBEDTLS_MEMORY_BUFFER_ALLOC_C) -#error "MBEDTLS_MEMORY_BACKTRACE defined, but not all prerequesites" -#endif - -#if defined(MBEDTLS_MEMORY_DEBUG) && !defined(MBEDTLS_MEMORY_BUFFER_ALLOC_C) -#error "MBEDTLS_MEMORY_DEBUG defined, but not all prerequesites" -#endif - -#if defined(MBEDTLS_PADLOCK_C) && !defined(MBEDTLS_HAVE_ASM) -#error "MBEDTLS_PADLOCK_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PEM_PARSE_C) && !defined(MBEDTLS_BASE64_C) -#error "MBEDTLS_PEM_PARSE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PEM_WRITE_C) && !defined(MBEDTLS_BASE64_C) -#error "MBEDTLS_PEM_WRITE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PK_C) && \ - ( !defined(MBEDTLS_RSA_C) && !defined(MBEDTLS_ECP_C) ) -#error "MBEDTLS_PK_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PK_PARSE_C) && !defined(MBEDTLS_PK_C) -#error "MBEDTLS_PK_PARSE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PK_WRITE_C) && !defined(MBEDTLS_PK_C) -#error "MBEDTLS_PK_WRITE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_EXIT_ALT) && !defined(MBEDTLS_PLATFORM_C) -#error "MBEDTLS_PLATFORM_EXIT_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_EXIT_MACRO) && !defined(MBEDTLS_PLATFORM_C) -#error "MBEDTLS_PLATFORM_EXIT_MACRO defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_EXIT_MACRO) &&\ - ( defined(MBEDTLS_PLATFORM_STD_EXIT) ||\ - defined(MBEDTLS_PLATFORM_EXIT_ALT) ) -#error "MBEDTLS_PLATFORM_EXIT_MACRO and MBEDTLS_PLATFORM_STD_EXIT/MBEDTLS_PLATFORM_EXIT_ALT cannot be defined simultaneously" -#endif - -#if defined(MBEDTLS_PLATFORM_TIME_ALT) &&\ - ( !defined(MBEDTLS_PLATFORM_C) ||\ - !defined(MBEDTLS_HAVE_TIME) ) -#error "MBEDTLS_PLATFORM_TIME_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_TIME_MACRO) &&\ - ( !defined(MBEDTLS_PLATFORM_C) ||\ - !defined(MBEDTLS_HAVE_TIME) ) -#error "MBEDTLS_PLATFORM_TIME_MACRO defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_TIME_TYPE_MACRO) &&\ - ( !defined(MBEDTLS_PLATFORM_C) ||\ - !defined(MBEDTLS_HAVE_TIME) ) -#error "MBEDTLS_PLATFORM_TIME_TYPE_MACRO defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_TIME_MACRO) &&\ - ( defined(MBEDTLS_PLATFORM_STD_TIME) ||\ - defined(MBEDTLS_PLATFORM_TIME_ALT) ) -#error "MBEDTLS_PLATFORM_TIME_MACRO and MBEDTLS_PLATFORM_STD_TIME/MBEDTLS_PLATFORM_TIME_ALT cannot be defined simultaneously" -#endif - -#if defined(MBEDTLS_PLATFORM_TIME_TYPE_MACRO) &&\ - ( defined(MBEDTLS_PLATFORM_STD_TIME) ||\ - defined(MBEDTLS_PLATFORM_TIME_ALT) ) -#error "MBEDTLS_PLATFORM_TIME_TYPE_MACRO and MBEDTLS_PLATFORM_STD_TIME/MBEDTLS_PLATFORM_TIME_ALT cannot be defined simultaneously" -#endif - -#if defined(MBEDTLS_PLATFORM_FPRINTF_ALT) && !defined(MBEDTLS_PLATFORM_C) -#error "MBEDTLS_PLATFORM_FPRINTF_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_FPRINTF_MACRO) && !defined(MBEDTLS_PLATFORM_C) -#error "MBEDTLS_PLATFORM_FPRINTF_MACRO defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_FPRINTF_MACRO) &&\ - ( defined(MBEDTLS_PLATFORM_STD_FPRINTF) ||\ - defined(MBEDTLS_PLATFORM_FPRINTF_ALT) ) -#error "MBEDTLS_PLATFORM_FPRINTF_MACRO and MBEDTLS_PLATFORM_STD_FPRINTF/MBEDTLS_PLATFORM_FPRINTF_ALT cannot be defined simultaneously" -#endif - -#if defined(MBEDTLS_PLATFORM_FREE_MACRO) &&\ - ( !defined(MBEDTLS_PLATFORM_C) || !defined(MBEDTLS_PLATFORM_MEMORY) ) -#error "MBEDTLS_PLATFORM_FREE_MACRO defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_FREE_MACRO) &&\ - defined(MBEDTLS_PLATFORM_STD_FREE) -#error "MBEDTLS_PLATFORM_FREE_MACRO and MBEDTLS_PLATFORM_STD_FREE cannot be defined simultaneously" -#endif - -#if defined(MBEDTLS_PLATFORM_FREE_MACRO) && !defined(MBEDTLS_PLATFORM_CALLOC_MACRO) -#error "MBEDTLS_PLATFORM_CALLOC_MACRO must be defined if MBEDTLS_PLATFORM_FREE_MACRO is" -#endif - -#if defined(MBEDTLS_PLATFORM_CALLOC_MACRO) &&\ - ( !defined(MBEDTLS_PLATFORM_C) || !defined(MBEDTLS_PLATFORM_MEMORY) ) -#error "MBEDTLS_PLATFORM_CALLOC_MACRO defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_CALLOC_MACRO) &&\ - defined(MBEDTLS_PLATFORM_STD_CALLOC) -#error "MBEDTLS_PLATFORM_CALLOC_MACRO and MBEDTLS_PLATFORM_STD_CALLOC cannot be defined simultaneously" -#endif - -#if defined(MBEDTLS_PLATFORM_CALLOC_MACRO) && !defined(MBEDTLS_PLATFORM_FREE_MACRO) -#error "MBEDTLS_PLATFORM_FREE_MACRO must be defined if MBEDTLS_PLATFORM_CALLOC_MACRO is" -#endif - -#if defined(MBEDTLS_PLATFORM_MEMORY) && !defined(MBEDTLS_PLATFORM_C) -#error "MBEDTLS_PLATFORM_MEMORY defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_PRINTF_ALT) && !defined(MBEDTLS_PLATFORM_C) -#error "MBEDTLS_PLATFORM_PRINTF_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_PRINTF_MACRO) && !defined(MBEDTLS_PLATFORM_C) -#error "MBEDTLS_PLATFORM_PRINTF_MACRO defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_PRINTF_MACRO) &&\ - ( defined(MBEDTLS_PLATFORM_STD_PRINTF) ||\ - defined(MBEDTLS_PLATFORM_PRINTF_ALT) ) -#error "MBEDTLS_PLATFORM_PRINTF_MACRO and MBEDTLS_PLATFORM_STD_PRINTF/MBEDTLS_PLATFORM_PRINTF_ALT cannot be defined simultaneously" -#endif - -#if defined(MBEDTLS_PLATFORM_SNPRINTF_ALT) && !defined(MBEDTLS_PLATFORM_C) -#error "MBEDTLS_PLATFORM_SNPRINTF_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_SNPRINTF_MACRO) && !defined(MBEDTLS_PLATFORM_C) -#error "MBEDTLS_PLATFORM_SNPRINTF_MACRO defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_SNPRINTF_MACRO) &&\ - ( defined(MBEDTLS_PLATFORM_STD_SNPRINTF) ||\ - defined(MBEDTLS_PLATFORM_SNPRINTF_ALT) ) -#error "MBEDTLS_PLATFORM_SNPRINTF_MACRO and MBEDTLS_PLATFORM_STD_SNPRINTF/MBEDTLS_PLATFORM_SNPRINTF_ALT cannot be defined simultaneously" -#endif - -#if defined(MBEDTLS_PLATFORM_STD_MEM_HDR) &&\ - !defined(MBEDTLS_PLATFORM_NO_STD_FUNCTIONS) -#error "MBEDTLS_PLATFORM_STD_MEM_HDR defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_STD_CALLOC) && !defined(MBEDTLS_PLATFORM_MEMORY) -#error "MBEDTLS_PLATFORM_STD_CALLOC defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_STD_CALLOC) && !defined(MBEDTLS_PLATFORM_MEMORY) -#error "MBEDTLS_PLATFORM_STD_CALLOC defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_STD_FREE) && !defined(MBEDTLS_PLATFORM_MEMORY) -#error "MBEDTLS_PLATFORM_STD_FREE defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_STD_EXIT) &&\ - !defined(MBEDTLS_PLATFORM_EXIT_ALT) -#error "MBEDTLS_PLATFORM_STD_EXIT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_STD_TIME) &&\ - ( !defined(MBEDTLS_PLATFORM_TIME_ALT) ||\ - !defined(MBEDTLS_HAVE_TIME) ) -#error "MBEDTLS_PLATFORM_STD_TIME defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_STD_FPRINTF) &&\ - !defined(MBEDTLS_PLATFORM_FPRINTF_ALT) -#error "MBEDTLS_PLATFORM_STD_FPRINTF defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_STD_PRINTF) &&\ - !defined(MBEDTLS_PLATFORM_PRINTF_ALT) -#error "MBEDTLS_PLATFORM_STD_PRINTF defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_STD_SNPRINTF) &&\ - !defined(MBEDTLS_PLATFORM_SNPRINTF_ALT) -#error "MBEDTLS_PLATFORM_STD_SNPRINTF defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_ENTROPY_NV_SEED) &&\ - ( !defined(MBEDTLS_PLATFORM_C) || !defined(MBEDTLS_ENTROPY_C) ) -#error "MBEDTLS_ENTROPY_NV_SEED defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_NV_SEED_ALT) &&\ - !defined(MBEDTLS_ENTROPY_NV_SEED) -#error "MBEDTLS_PLATFORM_NV_SEED_ALT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_STD_NV_SEED_READ) &&\ - !defined(MBEDTLS_PLATFORM_NV_SEED_ALT) -#error "MBEDTLS_PLATFORM_STD_NV_SEED_READ defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_STD_NV_SEED_WRITE) &&\ - !defined(MBEDTLS_PLATFORM_NV_SEED_ALT) -#error "MBEDTLS_PLATFORM_STD_NV_SEED_WRITE defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PLATFORM_NV_SEED_READ_MACRO) &&\ - ( defined(MBEDTLS_PLATFORM_STD_NV_SEED_READ) ||\ - defined(MBEDTLS_PLATFORM_NV_SEED_ALT) ) -#error "MBEDTLS_PLATFORM_NV_SEED_READ_MACRO and MBEDTLS_PLATFORM_STD_NV_SEED_READ cannot be defined simultaneously" -#endif - -#if defined(MBEDTLS_PLATFORM_NV_SEED_WRITE_MACRO) &&\ - ( defined(MBEDTLS_PLATFORM_STD_NV_SEED_WRITE) ||\ - defined(MBEDTLS_PLATFORM_NV_SEED_ALT) ) -#error "MBEDTLS_PLATFORM_NV_SEED_WRITE_MACRO and MBEDTLS_PLATFORM_STD_NV_SEED_WRITE cannot be defined simultaneously" -#endif - -#if defined(MBEDTLS_PSA_CRYPTO_C) && \ - !( ( ( defined(MBEDTLS_CTR_DRBG_C) || defined(MBEDTLS_HMAC_DRBG_C) ) && \ - defined(MBEDTLS_ENTROPY_C) ) || \ - defined(MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG) ) -#error "MBEDTLS_PSA_CRYPTO_C defined, but not all prerequisites (missing RNG)" -#endif - -#if defined(MBEDTLS_PSA_CRYPTO_SPM) && !defined(MBEDTLS_PSA_CRYPTO_C) -#error "MBEDTLS_PSA_CRYPTO_SPM defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PSA_CRYPTO_SE_C) && \ - ! ( defined(MBEDTLS_PSA_CRYPTO_C) && \ - defined(MBEDTLS_PSA_CRYPTO_STORAGE_C) ) -#error "MBEDTLS_PSA_CRYPTO_SE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PSA_CRYPTO_STORAGE_C) && \ - ! defined(MBEDTLS_PSA_CRYPTO_C) -#error "MBEDTLS_PSA_CRYPTO_STORAGE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PSA_INJECT_ENTROPY) && \ - !( defined(MBEDTLS_PSA_CRYPTO_STORAGE_C) && \ - defined(MBEDTLS_ENTROPY_NV_SEED) ) -#error "MBEDTLS_PSA_INJECT_ENTROPY defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PSA_INJECT_ENTROPY) && \ - !defined(MBEDTLS_NO_DEFAULT_ENTROPY_SOURCES) -#error "MBEDTLS_PSA_INJECT_ENTROPY is not compatible with actual entropy sources" -#endif - -#if defined(MBEDTLS_PSA_INJECT_ENTROPY) && \ - defined(MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG) -#error "MBEDTLS_PSA_INJECT_ENTROPY is not compatible with MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG" -#endif - -#if defined(MBEDTLS_PSA_ITS_FILE_C) && \ - !defined(MBEDTLS_FS_IO) -#error "MBEDTLS_PSA_ITS_FILE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_PSA_CRYPTO_KEY_ID_ENCODES_OWNER) && \ - defined(MBEDTLS_USE_PSA_CRYPTO) -#error "MBEDTLS_PSA_CRYPTO_KEY_ID_ENCODES_OWNER defined, but it cannot coexist with MBEDTLS_USE_PSA_CRYPTO." -#endif - -#if defined(MBEDTLS_RSA_C) && ( !defined(MBEDTLS_BIGNUM_C) || \ - !defined(MBEDTLS_OID_C) ) -#error "MBEDTLS_RSA_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_RSA_C) && ( !defined(MBEDTLS_PKCS1_V21) && \ - !defined(MBEDTLS_PKCS1_V15) ) -#error "MBEDTLS_RSA_C defined, but none of the PKCS1 versions enabled" -#endif - -#if defined(MBEDTLS_X509_RSASSA_PSS_SUPPORT) && \ - ( !defined(MBEDTLS_RSA_C) || !defined(MBEDTLS_PKCS1_V21) ) -#error "MBEDTLS_X509_RSASSA_PSS_SUPPORT defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SHA384_C) && !defined(MBEDTLS_SHA512_C) -#error "MBEDTLS_SHA384_C defined without MBEDTLS_SHA512_C" -#endif - -#if defined(MBEDTLS_SHA224_C) && !defined(MBEDTLS_SHA256_C) -#error "MBEDTLS_SHA224_C defined without MBEDTLS_SHA256_C" -#endif - -#if defined(MBEDTLS_SHA256_C) && !defined(MBEDTLS_SHA224_C) -#error "MBEDTLS_SHA256_C defined without MBEDTLS_SHA224_C" -#endif - -#if defined(MBEDTLS_SSL_PROTO_TLS1_2) && ( !defined(MBEDTLS_SHA1_C) && \ - !defined(MBEDTLS_SHA256_C) && !defined(MBEDTLS_SHA512_C) ) -#error "MBEDTLS_SSL_PROTO_TLS1_2 defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_PROTO_TLS1_3_EXPERIMENTAL) && ( !defined(MBEDTLS_HKDF_C) && \ - !defined(MBEDTLS_SHA256_C) && !defined(MBEDTLS_SHA512_C) ) -#error "MBEDTLS_SSL_PROTO_TLS1_3_EXPERIMENTAL defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_PROTO_TLS1_2) && \ - !(defined(MBEDTLS_KEY_EXCHANGE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_DHE_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED) ) -#error "One or more versions of the TLS protocol are enabled " \ - "but no key exchange methods defined with MBEDTLS_KEY_EXCHANGE_xxxx" -#endif - -#if defined(MBEDTLS_SSL_PROTO_DTLS) && \ - !defined(MBEDTLS_SSL_PROTO_TLS1_2) -#error "MBEDTLS_SSL_PROTO_DTLS defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_CLI_C) && !defined(MBEDTLS_SSL_TLS_C) -#error "MBEDTLS_SSL_CLI_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_TLS_C) && ( !defined(MBEDTLS_CIPHER_C) || \ - !defined(MBEDTLS_MD_C) ) -#error "MBEDTLS_SSL_TLS_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_SRV_C) && !defined(MBEDTLS_SSL_TLS_C) -#error "MBEDTLS_SSL_SRV_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_TLS_C) && !defined(MBEDTLS_SSL_PROTO_TLS1_2) -#error "MBEDTLS_SSL_TLS_C defined, but no protocols are active" -#endif - -#if defined(MBEDTLS_SSL_DTLS_HELLO_VERIFY) && !defined(MBEDTLS_SSL_PROTO_DTLS) -#error "MBEDTLS_SSL_DTLS_HELLO_VERIFY defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_DTLS_CLIENT_PORT_REUSE) && \ - !defined(MBEDTLS_SSL_DTLS_HELLO_VERIFY) -#error "MBEDTLS_SSL_DTLS_CLIENT_PORT_REUSE defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_DTLS_ANTI_REPLAY) && \ - ( !defined(MBEDTLS_SSL_TLS_C) || !defined(MBEDTLS_SSL_PROTO_DTLS) ) -#error "MBEDTLS_SSL_DTLS_ANTI_REPLAY defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_DTLS_CONNECTION_ID) && \ - ( !defined(MBEDTLS_SSL_TLS_C) || !defined(MBEDTLS_SSL_PROTO_DTLS) ) -#error "MBEDTLS_SSL_DTLS_CONNECTION_ID defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_DTLS_CONNECTION_ID) && \ - defined(MBEDTLS_SSL_CID_IN_LEN_MAX) && \ - MBEDTLS_SSL_CID_IN_LEN_MAX > 255 -#error "MBEDTLS_SSL_CID_IN_LEN_MAX too large (max 255)" -#endif - -#if defined(MBEDTLS_SSL_DTLS_CONNECTION_ID) && \ - defined(MBEDTLS_SSL_CID_OUT_LEN_MAX) && \ - MBEDTLS_SSL_CID_OUT_LEN_MAX > 255 -#error "MBEDTLS_SSL_CID_OUT_LEN_MAX too large (max 255)" -#endif - -#if defined(MBEDTLS_SSL_ENCRYPT_THEN_MAC) && \ - !defined(MBEDTLS_SSL_PROTO_TLS1_2) -#error "MBEDTLS_SSL_ENCRYPT_THEN_MAC defined, but not all prerequsites" -#endif - -#if defined(MBEDTLS_SSL_EXTENDED_MASTER_SECRET) && \ - !defined(MBEDTLS_SSL_PROTO_TLS1_2) -#error "MBEDTLS_SSL_EXTENDED_MASTER_SECRET defined, but not all prerequsites" -#endif - -#if defined(MBEDTLS_SSL_TICKET_C) && !defined(MBEDTLS_CIPHER_C) -#error "MBEDTLS_SSL_TICKET_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_SERVER_NAME_INDICATION) && \ - !defined(MBEDTLS_X509_CRT_PARSE_C) -#error "MBEDTLS_SSL_SERVER_NAME_INDICATION defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_THREADING_PTHREAD) -#if !defined(MBEDTLS_THREADING_C) || defined(MBEDTLS_THREADING_IMPL) -#error "MBEDTLS_THREADING_PTHREAD defined, but not all prerequisites" -#endif -#define MBEDTLS_THREADING_IMPL -#endif - -#if defined(MBEDTLS_THREADING_ALT) -#if !defined(MBEDTLS_THREADING_C) || defined(MBEDTLS_THREADING_IMPL) -#error "MBEDTLS_THREADING_ALT defined, but not all prerequisites" -#endif -#define MBEDTLS_THREADING_IMPL -#endif - -#if defined(MBEDTLS_THREADING_C) && !defined(MBEDTLS_THREADING_IMPL) -#error "MBEDTLS_THREADING_C defined, single threading implementation required" -#endif -#undef MBEDTLS_THREADING_IMPL - -#if defined(MBEDTLS_USE_PSA_CRYPTO) && !defined(MBEDTLS_PSA_CRYPTO_C) -#error "MBEDTLS_USE_PSA_CRYPTO defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_VERSION_FEATURES) && !defined(MBEDTLS_VERSION_C) -#error "MBEDTLS_VERSION_FEATURES defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_X509_USE_C) && ( !defined(MBEDTLS_BIGNUM_C) || \ - !defined(MBEDTLS_OID_C) || !defined(MBEDTLS_ASN1_PARSE_C) || \ - !defined(MBEDTLS_PK_PARSE_C) ) -#error "MBEDTLS_X509_USE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_X509_CREATE_C) && ( !defined(MBEDTLS_BIGNUM_C) || \ - !defined(MBEDTLS_OID_C) || !defined(MBEDTLS_ASN1_WRITE_C) || \ - !defined(MBEDTLS_PK_WRITE_C) ) -#error "MBEDTLS_X509_CREATE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_X509_CRT_PARSE_C) && ( !defined(MBEDTLS_X509_USE_C) ) -#error "MBEDTLS_X509_CRT_PARSE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_X509_CRL_PARSE_C) && ( !defined(MBEDTLS_X509_USE_C) ) -#error "MBEDTLS_X509_CRL_PARSE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_X509_CSR_PARSE_C) && ( !defined(MBEDTLS_X509_USE_C) ) -#error "MBEDTLS_X509_CSR_PARSE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_X509_CRT_WRITE_C) && ( !defined(MBEDTLS_X509_CREATE_C) ) -#error "MBEDTLS_X509_CRT_WRITE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_X509_CSR_WRITE_C) && ( !defined(MBEDTLS_X509_CREATE_C) ) -#error "MBEDTLS_X509_CSR_WRITE_C defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_HAVE_INT32) && defined(MBEDTLS_HAVE_INT64) -#error "MBEDTLS_HAVE_INT32 and MBEDTLS_HAVE_INT64 cannot be defined simultaneously" -#endif /* MBEDTLS_HAVE_INT32 && MBEDTLS_HAVE_INT64 */ - -#if ( defined(MBEDTLS_HAVE_INT32) || defined(MBEDTLS_HAVE_INT64) ) && \ - defined(MBEDTLS_HAVE_ASM) -#error "MBEDTLS_HAVE_INT32/MBEDTLS_HAVE_INT64 and MBEDTLS_HAVE_ASM cannot be defined simultaneously" -#endif /* (MBEDTLS_HAVE_INT32 || MBEDTLS_HAVE_INT64) && MBEDTLS_HAVE_ASM */ - -#if defined(MBEDTLS_SSL_DTLS_SRTP) && ( !defined(MBEDTLS_SSL_PROTO_DTLS) ) -#error "MBEDTLS_SSL_DTLS_SRTP defined, but not all prerequisites" -#endif - -#if defined(MBEDTLS_SSL_VARIABLE_BUFFER_LENGTH) && ( !defined(MBEDTLS_SSL_MAX_FRAGMENT_LENGTH) ) -#error "MBEDTLS_SSL_VARIABLE_BUFFER_LENGTH defined, but not all prerequisites" -#endif - - - -/* Reject attempts to enable options that have been removed and that could - * cause a build to succeed but with features removed. */ - -#if defined(MBEDTLS_HAVEGE_C) //no-check-names -#error "MBEDTLS_HAVEGE_C was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/2599" -#endif - -#if defined(MBEDTLS_SSL_HW_RECORD_ACCEL) //no-check-names -#error "MBEDTLS_SSL_HW_RECORD_ACCEL was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/4031" -#endif - -#if defined(MBEDTLS_SSL_PROTO_SSL3) //no-check-names -#error "MBEDTLS_SSL_PROTO_SSL3 (SSL v3.0 support) was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/4031" -#endif - -#if defined(MBEDTLS_SSL_SRV_SUPPORT_SSLV2_CLIENT_HELLO) //no-check-names -#error "MBEDTLS_SSL_SRV_SUPPORT_SSLV2_CLIENT_HELLO (SSL v2 ClientHello support) was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/4031" -#endif - -#if defined(MBEDTLS_SSL_TRUNCATED_HMAC_COMPAT) //no-check-names -#error "MBEDTLS_SSL_TRUNCATED_HMAC_COMPAT (compatibility with the buggy implementation of truncated HMAC in Mbed TLS up to 2.7) was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/4031" -#endif - -#if defined(MBEDTLS_TLS_DEFAULT_ALLOW_SHA1_IN_CERTIFICATES) //no-check-names -#error "MBEDTLS_TLS_DEFAULT_ALLOW_SHA1_IN_CERTIFICATES was removed in Mbed TLS 3.0. See the ChangeLog entry if you really need SHA-1-signed certificates." -#endif - -#if defined(MBEDTLS_ZLIB_SUPPORT) //no-check-names -#error "MBEDTLS_ZLIB_SUPPORT was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/4031" -#endif - -#if defined(MBEDTLS_SSL_PROTO_TLS1) //no-check-names -#error "MBEDTLS_SSL_PROTO_TLS1 (TLS v1.0 support) was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/4286" -#endif - -#if defined(MBEDTLS_SSL_PROTO_TLS1_1) //no-check-names -#error "MBEDTLS_SSL_PROTO_TLS1_1 (TLS v1.1 support) was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/4286" -#endif - -#if defined(MBEDTLS_CHECK_PARAMS) //no-check-names -#error "MBEDTLS_CHECK_PARAMS was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/4313" -#endif - -#if defined(MBEDTLS_SSL_CID_PADDING_GRANULARITY) //no-check-names -#error "MBEDTLS_SSL_CID_PADDING_GRANULARITY was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/4335" -#endif - -#if defined(MBEDTLS_SSL_TLS1_3_PADDING_GRANULARITY) //no-check-names -#error "MBEDTLS_SSL_TLS1_3_PADDING_GRANULARITY was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/4335" -#endif - -#if defined(MBEDTLS_SSL_TRUNCATED_HMAC) //no-check-names -#error "MBEDTLS_SSL_TRUNCATED_HMAC was removed in Mbed TLS 3.0. See https://github.com/ARMmbed/mbedtls/issues/4341" -#endif - -/* - * Avoid warning from -pedantic. This is a convenient place for this - * workaround since this is included by every single file before the - * #if defined(MBEDTLS_xxx_C) that results in empty translation units. - */ -typedef int mbedtls_iso_c_forbids_empty_translation_units; - -#endif /* MBEDTLS_CHECK_CONFIG_H */ diff --git a/windows/mbedtls/cipher.h b/windows/mbedtls/cipher.h deleted file mode 100644 index 7afdc6ac..00000000 --- a/windows/mbedtls/cipher.h +++ /dev/null @@ -1,983 +0,0 @@ -/** - * \file cipher.h - * - * \brief This file contains an abstraction interface for use with the cipher - * primitives provided by the library. It provides a common interface to all of - * the available cipher operations. - * - * \author Adriaan de Jong - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_CIPHER_H -#define MBEDTLS_CIPHER_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include "mbedtls/platform_util.h" - -#if defined(MBEDTLS_GCM_C) || defined(MBEDTLS_CCM_C) || defined(MBEDTLS_CHACHAPOLY_C) -#define MBEDTLS_CIPHER_MODE_AEAD -#endif - -#if defined(MBEDTLS_CIPHER_MODE_CBC) -#define MBEDTLS_CIPHER_MODE_WITH_PADDING -#endif - -#if defined(MBEDTLS_CIPHER_NULL_CIPHER) || \ - defined(MBEDTLS_CHACHA20_C) -#define MBEDTLS_CIPHER_MODE_STREAM -#endif - -#if ( defined(__ARMCC_VERSION) || defined(_MSC_VER) ) && \ - !defined(inline) && !defined(__cplusplus) -#define inline __inline -#endif - -#define MBEDTLS_ERR_CIPHER_FEATURE_UNAVAILABLE -0x6080 /**< The selected feature is not available. */ -#define MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA -0x6100 /**< Bad input parameters. */ -#define MBEDTLS_ERR_CIPHER_ALLOC_FAILED -0x6180 /**< Failed to allocate memory. */ -#define MBEDTLS_ERR_CIPHER_INVALID_PADDING -0x6200 /**< Input data contains invalid padding and is rejected. */ -#define MBEDTLS_ERR_CIPHER_FULL_BLOCK_EXPECTED -0x6280 /**< Decryption of block requires a full block. */ -#define MBEDTLS_ERR_CIPHER_AUTH_FAILED -0x6300 /**< Authentication failed (for AEAD modes). */ -#define MBEDTLS_ERR_CIPHER_INVALID_CONTEXT -0x6380 /**< The context is invalid. For example, because it was freed. */ - -#define MBEDTLS_CIPHER_VARIABLE_IV_LEN 0x01 /**< Cipher accepts IVs of variable length. */ -#define MBEDTLS_CIPHER_VARIABLE_KEY_LEN 0x02 /**< Cipher accepts keys of variable length. */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief Supported cipher types. - * - * \warning DES is considered weak cipher and its use - * constitutes a security risk. Arm recommends considering stronger - * ciphers instead. - */ -typedef enum { - MBEDTLS_CIPHER_ID_NONE = 0, /**< Placeholder to mark the end of cipher ID lists. */ - MBEDTLS_CIPHER_ID_NULL, /**< The identity cipher, treated as a stream cipher. */ - MBEDTLS_CIPHER_ID_AES, /**< The AES cipher. */ - MBEDTLS_CIPHER_ID_DES, /**< The DES cipher. */ - MBEDTLS_CIPHER_ID_3DES, /**< The Triple DES cipher. */ - MBEDTLS_CIPHER_ID_CAMELLIA, /**< The Camellia cipher. */ - MBEDTLS_CIPHER_ID_ARIA, /**< The Aria cipher. */ - MBEDTLS_CIPHER_ID_CHACHA20, /**< The ChaCha20 cipher. */ -} mbedtls_cipher_id_t; - -/** - * \brief Supported {cipher type, cipher mode} pairs. - * - * \warning DES is considered weak cipher and its use - * constitutes a security risk. Arm recommends considering stronger - * ciphers instead. - */ -typedef enum { - MBEDTLS_CIPHER_NONE = 0, /**< Placeholder to mark the end of cipher-pair lists. */ - MBEDTLS_CIPHER_NULL, /**< The identity stream cipher. */ - MBEDTLS_CIPHER_AES_128_ECB, /**< AES cipher with 128-bit ECB mode. */ - MBEDTLS_CIPHER_AES_192_ECB, /**< AES cipher with 192-bit ECB mode. */ - MBEDTLS_CIPHER_AES_256_ECB, /**< AES cipher with 256-bit ECB mode. */ - MBEDTLS_CIPHER_AES_128_CBC, /**< AES cipher with 128-bit CBC mode. */ - MBEDTLS_CIPHER_AES_192_CBC, /**< AES cipher with 192-bit CBC mode. */ - MBEDTLS_CIPHER_AES_256_CBC, /**< AES cipher with 256-bit CBC mode. */ - MBEDTLS_CIPHER_AES_128_CFB128, /**< AES cipher with 128-bit CFB128 mode. */ - MBEDTLS_CIPHER_AES_192_CFB128, /**< AES cipher with 192-bit CFB128 mode. */ - MBEDTLS_CIPHER_AES_256_CFB128, /**< AES cipher with 256-bit CFB128 mode. */ - MBEDTLS_CIPHER_AES_128_CTR, /**< AES cipher with 128-bit CTR mode. */ - MBEDTLS_CIPHER_AES_192_CTR, /**< AES cipher with 192-bit CTR mode. */ - MBEDTLS_CIPHER_AES_256_CTR, /**< AES cipher with 256-bit CTR mode. */ - MBEDTLS_CIPHER_AES_128_GCM, /**< AES cipher with 128-bit GCM mode. */ - MBEDTLS_CIPHER_AES_192_GCM, /**< AES cipher with 192-bit GCM mode. */ - MBEDTLS_CIPHER_AES_256_GCM, /**< AES cipher with 256-bit GCM mode. */ - MBEDTLS_CIPHER_CAMELLIA_128_ECB, /**< Camellia cipher with 128-bit ECB mode. */ - MBEDTLS_CIPHER_CAMELLIA_192_ECB, /**< Camellia cipher with 192-bit ECB mode. */ - MBEDTLS_CIPHER_CAMELLIA_256_ECB, /**< Camellia cipher with 256-bit ECB mode. */ - MBEDTLS_CIPHER_CAMELLIA_128_CBC, /**< Camellia cipher with 128-bit CBC mode. */ - MBEDTLS_CIPHER_CAMELLIA_192_CBC, /**< Camellia cipher with 192-bit CBC mode. */ - MBEDTLS_CIPHER_CAMELLIA_256_CBC, /**< Camellia cipher with 256-bit CBC mode. */ - MBEDTLS_CIPHER_CAMELLIA_128_CFB128, /**< Camellia cipher with 128-bit CFB128 mode. */ - MBEDTLS_CIPHER_CAMELLIA_192_CFB128, /**< Camellia cipher with 192-bit CFB128 mode. */ - MBEDTLS_CIPHER_CAMELLIA_256_CFB128, /**< Camellia cipher with 256-bit CFB128 mode. */ - MBEDTLS_CIPHER_CAMELLIA_128_CTR, /**< Camellia cipher with 128-bit CTR mode. */ - MBEDTLS_CIPHER_CAMELLIA_192_CTR, /**< Camellia cipher with 192-bit CTR mode. */ - MBEDTLS_CIPHER_CAMELLIA_256_CTR, /**< Camellia cipher with 256-bit CTR mode. */ - MBEDTLS_CIPHER_CAMELLIA_128_GCM, /**< Camellia cipher with 128-bit GCM mode. */ - MBEDTLS_CIPHER_CAMELLIA_192_GCM, /**< Camellia cipher with 192-bit GCM mode. */ - MBEDTLS_CIPHER_CAMELLIA_256_GCM, /**< Camellia cipher with 256-bit GCM mode. */ - MBEDTLS_CIPHER_DES_ECB, /**< DES cipher with ECB mode. */ - MBEDTLS_CIPHER_DES_CBC, /**< DES cipher with CBC mode. */ - MBEDTLS_CIPHER_DES_EDE_ECB, /**< DES cipher with EDE ECB mode. */ - MBEDTLS_CIPHER_DES_EDE_CBC, /**< DES cipher with EDE CBC mode. */ - MBEDTLS_CIPHER_DES_EDE3_ECB, /**< DES cipher with EDE3 ECB mode. */ - MBEDTLS_CIPHER_DES_EDE3_CBC, /**< DES cipher with EDE3 CBC mode. */ - MBEDTLS_CIPHER_AES_128_CCM, /**< AES cipher with 128-bit CCM mode. */ - MBEDTLS_CIPHER_AES_192_CCM, /**< AES cipher with 192-bit CCM mode. */ - MBEDTLS_CIPHER_AES_256_CCM, /**< AES cipher with 256-bit CCM mode. */ - MBEDTLS_CIPHER_CAMELLIA_128_CCM, /**< Camellia cipher with 128-bit CCM mode. */ - MBEDTLS_CIPHER_CAMELLIA_192_CCM, /**< Camellia cipher with 192-bit CCM mode. */ - MBEDTLS_CIPHER_CAMELLIA_256_CCM, /**< Camellia cipher with 256-bit CCM mode. */ - MBEDTLS_CIPHER_ARIA_128_ECB, /**< Aria cipher with 128-bit key and ECB mode. */ - MBEDTLS_CIPHER_ARIA_192_ECB, /**< Aria cipher with 192-bit key and ECB mode. */ - MBEDTLS_CIPHER_ARIA_256_ECB, /**< Aria cipher with 256-bit key and ECB mode. */ - MBEDTLS_CIPHER_ARIA_128_CBC, /**< Aria cipher with 128-bit key and CBC mode. */ - MBEDTLS_CIPHER_ARIA_192_CBC, /**< Aria cipher with 192-bit key and CBC mode. */ - MBEDTLS_CIPHER_ARIA_256_CBC, /**< Aria cipher with 256-bit key and CBC mode. */ - MBEDTLS_CIPHER_ARIA_128_CFB128, /**< Aria cipher with 128-bit key and CFB-128 mode. */ - MBEDTLS_CIPHER_ARIA_192_CFB128, /**< Aria cipher with 192-bit key and CFB-128 mode. */ - MBEDTLS_CIPHER_ARIA_256_CFB128, /**< Aria cipher with 256-bit key and CFB-128 mode. */ - MBEDTLS_CIPHER_ARIA_128_CTR, /**< Aria cipher with 128-bit key and CTR mode. */ - MBEDTLS_CIPHER_ARIA_192_CTR, /**< Aria cipher with 192-bit key and CTR mode. */ - MBEDTLS_CIPHER_ARIA_256_CTR, /**< Aria cipher with 256-bit key and CTR mode. */ - MBEDTLS_CIPHER_ARIA_128_GCM, /**< Aria cipher with 128-bit key and GCM mode. */ - MBEDTLS_CIPHER_ARIA_192_GCM, /**< Aria cipher with 192-bit key and GCM mode. */ - MBEDTLS_CIPHER_ARIA_256_GCM, /**< Aria cipher with 256-bit key and GCM mode. */ - MBEDTLS_CIPHER_ARIA_128_CCM, /**< Aria cipher with 128-bit key and CCM mode. */ - MBEDTLS_CIPHER_ARIA_192_CCM, /**< Aria cipher with 192-bit key and CCM mode. */ - MBEDTLS_CIPHER_ARIA_256_CCM, /**< Aria cipher with 256-bit key and CCM mode. */ - MBEDTLS_CIPHER_AES_128_OFB, /**< AES 128-bit cipher in OFB mode. */ - MBEDTLS_CIPHER_AES_192_OFB, /**< AES 192-bit cipher in OFB mode. */ - MBEDTLS_CIPHER_AES_256_OFB, /**< AES 256-bit cipher in OFB mode. */ - MBEDTLS_CIPHER_AES_128_XTS, /**< AES 128-bit cipher in XTS block mode. */ - MBEDTLS_CIPHER_AES_256_XTS, /**< AES 256-bit cipher in XTS block mode. */ - MBEDTLS_CIPHER_CHACHA20, /**< ChaCha20 stream cipher. */ - MBEDTLS_CIPHER_CHACHA20_POLY1305, /**< ChaCha20-Poly1305 AEAD cipher. */ - MBEDTLS_CIPHER_AES_128_KW, /**< AES cipher with 128-bit NIST KW mode. */ - MBEDTLS_CIPHER_AES_192_KW, /**< AES cipher with 192-bit NIST KW mode. */ - MBEDTLS_CIPHER_AES_256_KW, /**< AES cipher with 256-bit NIST KW mode. */ - MBEDTLS_CIPHER_AES_128_KWP, /**< AES cipher with 128-bit NIST KWP mode. */ - MBEDTLS_CIPHER_AES_192_KWP, /**< AES cipher with 192-bit NIST KWP mode. */ - MBEDTLS_CIPHER_AES_256_KWP, /**< AES cipher with 256-bit NIST KWP mode. */ -} mbedtls_cipher_type_t; - -/** Supported cipher modes. */ -typedef enum { - MBEDTLS_MODE_NONE = 0, /**< None. */ - MBEDTLS_MODE_ECB, /**< The ECB cipher mode. */ - MBEDTLS_MODE_CBC, /**< The CBC cipher mode. */ - MBEDTLS_MODE_CFB, /**< The CFB cipher mode. */ - MBEDTLS_MODE_OFB, /**< The OFB cipher mode. */ - MBEDTLS_MODE_CTR, /**< The CTR cipher mode. */ - MBEDTLS_MODE_GCM, /**< The GCM cipher mode. */ - MBEDTLS_MODE_STREAM, /**< The stream cipher mode. */ - MBEDTLS_MODE_CCM, /**< The CCM cipher mode. */ - MBEDTLS_MODE_XTS, /**< The XTS cipher mode. */ - MBEDTLS_MODE_CHACHAPOLY, /**< The ChaCha-Poly cipher mode. */ - MBEDTLS_MODE_KW, /**< The SP800-38F KW mode */ - MBEDTLS_MODE_KWP, /**< The SP800-38F KWP mode */ -} mbedtls_cipher_mode_t; - -/** Supported cipher padding types. */ -typedef enum { - MBEDTLS_PADDING_PKCS7 = 0, /**< PKCS7 padding (default). */ - MBEDTLS_PADDING_ONE_AND_ZEROS, /**< ISO/IEC 7816-4 padding. */ - MBEDTLS_PADDING_ZEROS_AND_LEN, /**< ANSI X.923 padding. */ - MBEDTLS_PADDING_ZEROS, /**< Zero padding (not reversible). */ - MBEDTLS_PADDING_NONE, /**< Never pad (full blocks only). */ -} mbedtls_cipher_padding_t; - -/** Type of operation. */ -typedef enum { - MBEDTLS_OPERATION_NONE = -1, - MBEDTLS_DECRYPT = 0, - MBEDTLS_ENCRYPT, -} mbedtls_operation_t; - -enum { - /** Undefined key length. */ - MBEDTLS_KEY_LENGTH_NONE = 0, - /** Key length, in bits (including parity), for DES keys. */ - MBEDTLS_KEY_LENGTH_DES = 64, - /** Key length in bits, including parity, for DES in two-key EDE. */ - MBEDTLS_KEY_LENGTH_DES_EDE = 128, - /** Key length in bits, including parity, for DES in three-key EDE. */ - MBEDTLS_KEY_LENGTH_DES_EDE3 = 192, -}; - -/** Maximum length of any IV, in Bytes. */ -/* This should ideally be derived automatically from list of ciphers. - * This should be kept in sync with MBEDTLS_SSL_MAX_IV_LENGTH defined - * in library/ssl_misc.h. */ -#define MBEDTLS_MAX_IV_LENGTH 16 - -/** Maximum block size of any cipher, in Bytes. */ -/* This should ideally be derived automatically from list of ciphers. - * This should be kept in sync with MBEDTLS_SSL_MAX_BLOCK_LENGTH defined - * in library/ssl_misc.h. */ -#define MBEDTLS_MAX_BLOCK_LENGTH 16 - -/** Maximum key length, in Bytes. */ -/* This should ideally be derived automatically from list of ciphers. - * For now, only check whether XTS is enabled which uses 64 Byte keys, - * and use 32 Bytes as an upper bound for the maximum key length otherwise. - * This should be kept in sync with MBEDTLS_SSL_MAX_BLOCK_LENGTH defined - * in library/ssl_misc.h, which however deliberately ignores the case of XTS - * since the latter isn't used in SSL/TLS. */ -#if defined(MBEDTLS_CIPHER_MODE_XTS) -#define MBEDTLS_MAX_KEY_LENGTH 64 -#else -#define MBEDTLS_MAX_KEY_LENGTH 32 -#endif /* MBEDTLS_CIPHER_MODE_XTS */ - -/** - * Base cipher information (opaque struct). - */ -typedef struct mbedtls_cipher_base_t mbedtls_cipher_base_t; - -/** - * CMAC context (opaque struct). - */ -typedef struct mbedtls_cmac_context_t mbedtls_cmac_context_t; - -/** - * Cipher information. Allows calling cipher functions - * in a generic way. - */ -typedef struct mbedtls_cipher_info_t -{ - /** Full cipher identifier. For example, - * MBEDTLS_CIPHER_AES_256_CBC. - */ - mbedtls_cipher_type_t MBEDTLS_PRIVATE(type); - - /** The cipher mode. For example, MBEDTLS_MODE_CBC. */ - mbedtls_cipher_mode_t MBEDTLS_PRIVATE(mode); - - /** The cipher key length, in bits. This is the - * default length for variable sized ciphers. - * Includes parity bits for ciphers like DES. - */ - unsigned int MBEDTLS_PRIVATE(key_bitlen); - - /** Name of the cipher. */ - const char * MBEDTLS_PRIVATE(name); - - /** IV or nonce size, in Bytes. - * For ciphers that accept variable IV sizes, - * this is the recommended size. - */ - unsigned int MBEDTLS_PRIVATE(iv_size); - - /** Bitflag comprised of MBEDTLS_CIPHER_VARIABLE_IV_LEN and - * MBEDTLS_CIPHER_VARIABLE_KEY_LEN indicating whether the - * cipher supports variable IV or variable key sizes, respectively. - */ - int MBEDTLS_PRIVATE(flags); - - /** The block size, in Bytes. */ - unsigned int MBEDTLS_PRIVATE(block_size); - - /** Struct for base cipher information and functions. */ - const mbedtls_cipher_base_t *MBEDTLS_PRIVATE(base); - -} mbedtls_cipher_info_t; - -/** - * Generic cipher context. - */ -typedef struct mbedtls_cipher_context_t -{ - /** Information about the associated cipher. */ - const mbedtls_cipher_info_t *MBEDTLS_PRIVATE(cipher_info); - - /** Key length to use. */ - int MBEDTLS_PRIVATE(key_bitlen); - - /** Operation that the key of the context has been - * initialized for. - */ - mbedtls_operation_t MBEDTLS_PRIVATE(operation); - -#if defined(MBEDTLS_CIPHER_MODE_WITH_PADDING) - /** Padding functions to use, if relevant for - * the specific cipher mode. - */ - void (*MBEDTLS_PRIVATE(add_padding))( unsigned char *output, size_t olen, size_t data_len ); - int (*MBEDTLS_PRIVATE(get_padding))( unsigned char *input, size_t ilen, size_t *data_len ); -#endif - - /** Buffer for input that has not been processed yet. */ - unsigned char MBEDTLS_PRIVATE(unprocessed_data)[MBEDTLS_MAX_BLOCK_LENGTH]; - - /** Number of Bytes that have not been processed yet. */ - size_t MBEDTLS_PRIVATE(unprocessed_len); - - /** Current IV or NONCE_COUNTER for CTR-mode, data unit (or sector) number - * for XTS-mode. */ - unsigned char MBEDTLS_PRIVATE(iv)[MBEDTLS_MAX_IV_LENGTH]; - - /** IV size in Bytes, for ciphers with variable-length IVs. */ - size_t MBEDTLS_PRIVATE(iv_size); - - /** The cipher-specific context. */ - void *MBEDTLS_PRIVATE(cipher_ctx); - -#if defined(MBEDTLS_CMAC_C) - /** CMAC-specific context. */ - mbedtls_cmac_context_t *MBEDTLS_PRIVATE(cmac_ctx); -#endif - -#if defined(MBEDTLS_USE_PSA_CRYPTO) - /** Indicates whether the cipher operations should be performed - * by Mbed TLS' own crypto library or an external implementation - * of the PSA Crypto API. - * This is unset if the cipher context was established through - * mbedtls_cipher_setup(), and set if it was established through - * mbedtls_cipher_setup_psa(). - */ - unsigned char MBEDTLS_PRIVATE(psa_enabled); -#endif /* MBEDTLS_USE_PSA_CRYPTO */ - -} mbedtls_cipher_context_t; - -/** - * \brief This function retrieves the list of ciphers supported - * by the generic cipher module. - * - * For any cipher identifier in the returned list, you can - * obtain the corresponding generic cipher information structure - * via mbedtls_cipher_info_from_type(), which can then be used - * to prepare a cipher context via mbedtls_cipher_setup(). - * - * - * \return A statically-allocated array of cipher identifiers - * of type cipher_type_t. The last entry is zero. - */ -const int *mbedtls_cipher_list( void ); - -/** - * \brief This function retrieves the cipher-information - * structure associated with the given cipher name. - * - * \param cipher_name Name of the cipher to search for. This must not be - * \c NULL. - * - * \return The cipher information structure associated with the - * given \p cipher_name. - * \return \c NULL if the associated cipher information is not found. - */ -const mbedtls_cipher_info_t *mbedtls_cipher_info_from_string( const char *cipher_name ); - -/** - * \brief This function retrieves the cipher-information - * structure associated with the given cipher type. - * - * \param cipher_type Type of the cipher to search for. - * - * \return The cipher information structure associated with the - * given \p cipher_type. - * \return \c NULL if the associated cipher information is not found. - */ -const mbedtls_cipher_info_t *mbedtls_cipher_info_from_type( const mbedtls_cipher_type_t cipher_type ); - -/** - * \brief This function retrieves the cipher-information - * structure associated with the given cipher ID, - * key size and mode. - * - * \param cipher_id The ID of the cipher to search for. For example, - * #MBEDTLS_CIPHER_ID_AES. - * \param key_bitlen The length of the key in bits. - * \param mode The cipher mode. For example, #MBEDTLS_MODE_CBC. - * - * \return The cipher information structure associated with the - * given \p cipher_id. - * \return \c NULL if the associated cipher information is not found. - */ -const mbedtls_cipher_info_t *mbedtls_cipher_info_from_values( const mbedtls_cipher_id_t cipher_id, - int key_bitlen, - const mbedtls_cipher_mode_t mode ); - -/** - * \brief This function initializes a \p cipher_context as NONE. - * - * \param ctx The context to be initialized. This must not be \c NULL. - */ -void mbedtls_cipher_init( mbedtls_cipher_context_t *ctx ); - -/** - * \brief This function frees and clears the cipher-specific - * context of \p ctx. Freeing \p ctx itself remains the - * responsibility of the caller. - * - * \param ctx The context to be freed. If this is \c NULL, the - * function has no effect, otherwise this must point to an - * initialized context. - */ -void mbedtls_cipher_free( mbedtls_cipher_context_t *ctx ); - - -/** - * \brief This function prepares a cipher context for - * use with the given cipher primitive. - * - * \note After calling this function, you should call - * mbedtls_cipher_setkey() and, if the mode uses padding, - * mbedtls_cipher_set_padding_mode(), then for each - * message to encrypt or decrypt with this key, either: - * - mbedtls_cipher_crypt() for one-shot processing with - * non-AEAD modes; - * - mbedtls_cipher_auth_encrypt_ext() or - * mbedtls_cipher_auth_decrypt_ext() for one-shot - * processing with AEAD modes or NIST_KW; - * - for multi-part processing, see the documentation of - * mbedtls_cipher_reset(). - * - * \param ctx The context to prepare. This must be initialized by - * a call to mbedtls_cipher_init() first. - * \param cipher_info The cipher to use. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA on - * parameter-verification failure. - * \return #MBEDTLS_ERR_CIPHER_ALLOC_FAILED if allocation of the - * cipher-specific context fails. - */ -int mbedtls_cipher_setup( mbedtls_cipher_context_t *ctx, - const mbedtls_cipher_info_t *cipher_info ); - -#if defined(MBEDTLS_USE_PSA_CRYPTO) -/** - * \brief This function initializes a cipher context for - * PSA-based use with the given cipher primitive. - * - * \note See #MBEDTLS_USE_PSA_CRYPTO for information on PSA. - * - * \param ctx The context to initialize. May not be \c NULL. - * \param cipher_info The cipher to use. - * \param taglen For AEAD ciphers, the length in bytes of the - * authentication tag to use. Subsequent uses of - * mbedtls_cipher_auth_encrypt_ext() or - * mbedtls_cipher_auth_decrypt_ext() must provide - * the same tag length. - * For non-AEAD ciphers, the value must be \c 0. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA on - * parameter-verification failure. - * \return #MBEDTLS_ERR_CIPHER_ALLOC_FAILED if allocation of the - * cipher-specific context fails. - */ -int mbedtls_cipher_setup_psa( mbedtls_cipher_context_t *ctx, - const mbedtls_cipher_info_t *cipher_info, - size_t taglen ); -#endif /* MBEDTLS_USE_PSA_CRYPTO */ - -/** - * \brief This function returns the block size of the given cipher. - * - * \param ctx The context of the cipher. This must be initialized. - * - * \return The block size of the underlying cipher. - * \return \c 0 if \p ctx has not been initialized. - */ -static inline unsigned int mbedtls_cipher_get_block_size( - const mbedtls_cipher_context_t *ctx ) -{ - MBEDTLS_INTERNAL_VALIDATE_RET( ctx != NULL, 0 ); - if( ctx->MBEDTLS_PRIVATE(cipher_info) == NULL ) - return 0; - - return ctx->MBEDTLS_PRIVATE(cipher_info)->MBEDTLS_PRIVATE(block_size); -} - -/** - * \brief This function returns the mode of operation for - * the cipher. For example, MBEDTLS_MODE_CBC. - * - * \param ctx The context of the cipher. This must be initialized. - * - * \return The mode of operation. - * \return #MBEDTLS_MODE_NONE if \p ctx has not been initialized. - */ -static inline mbedtls_cipher_mode_t mbedtls_cipher_get_cipher_mode( - const mbedtls_cipher_context_t *ctx ) -{ - MBEDTLS_INTERNAL_VALIDATE_RET( ctx != NULL, MBEDTLS_MODE_NONE ); - if( ctx->MBEDTLS_PRIVATE(cipher_info) == NULL ) - return MBEDTLS_MODE_NONE; - - return ctx->MBEDTLS_PRIVATE(cipher_info)->MBEDTLS_PRIVATE(mode); -} - -/** - * \brief This function returns the size of the IV or nonce - * of the cipher, in Bytes. - * - * \param ctx The context of the cipher. This must be initialized. - * - * \return The recommended IV size if no IV has been set. - * \return \c 0 for ciphers not using an IV or a nonce. - * \return The actual size if an IV has been set. - */ -static inline int mbedtls_cipher_get_iv_size( - const mbedtls_cipher_context_t *ctx ) -{ - MBEDTLS_INTERNAL_VALIDATE_RET( ctx != NULL, 0 ); - if( ctx->MBEDTLS_PRIVATE(cipher_info) == NULL ) - return 0; - - if( ctx->MBEDTLS_PRIVATE(iv_size) != 0 ) - return (int) ctx->MBEDTLS_PRIVATE(iv_size); - - return (int) ctx->MBEDTLS_PRIVATE(cipher_info)->MBEDTLS_PRIVATE(iv_size); -} - -/** - * \brief This function returns the type of the given cipher. - * - * \param ctx The context of the cipher. This must be initialized. - * - * \return The type of the cipher. - * \return #MBEDTLS_CIPHER_NONE if \p ctx has not been initialized. - */ -static inline mbedtls_cipher_type_t mbedtls_cipher_get_type( - const mbedtls_cipher_context_t *ctx ) -{ - MBEDTLS_INTERNAL_VALIDATE_RET( - ctx != NULL, MBEDTLS_CIPHER_NONE ); - if( ctx->MBEDTLS_PRIVATE(cipher_info) == NULL ) - return MBEDTLS_CIPHER_NONE; - - return ctx->MBEDTLS_PRIVATE(cipher_info)->MBEDTLS_PRIVATE(type); -} - -/** - * \brief This function returns the name of the given cipher - * as a string. - * - * \param ctx The context of the cipher. This must be initialized. - * - * \return The name of the cipher. - * \return NULL if \p ctx has not been not initialized. - */ -static inline const char *mbedtls_cipher_get_name( - const mbedtls_cipher_context_t *ctx ) -{ - MBEDTLS_INTERNAL_VALIDATE_RET( ctx != NULL, 0 ); - if( ctx->MBEDTLS_PRIVATE(cipher_info) == NULL ) - return 0; - - return ctx->MBEDTLS_PRIVATE(cipher_info)->MBEDTLS_PRIVATE(name); -} - -/** - * \brief This function returns the key length of the cipher. - * - * \param ctx The context of the cipher. This must be initialized. - * - * \return The key length of the cipher in bits. - * \return #MBEDTLS_KEY_LENGTH_NONE if ctx \p has not been - * initialized. - */ -static inline int mbedtls_cipher_get_key_bitlen( - const mbedtls_cipher_context_t *ctx ) -{ - MBEDTLS_INTERNAL_VALIDATE_RET( - ctx != NULL, MBEDTLS_KEY_LENGTH_NONE ); - if( ctx->MBEDTLS_PRIVATE(cipher_info) == NULL ) - return MBEDTLS_KEY_LENGTH_NONE; - - return (int) ctx->MBEDTLS_PRIVATE(cipher_info)->MBEDTLS_PRIVATE(key_bitlen); -} - -/** - * \brief This function returns the operation of the given cipher. - * - * \param ctx The context of the cipher. This must be initialized. - * - * \return The type of operation: #MBEDTLS_ENCRYPT or #MBEDTLS_DECRYPT. - * \return #MBEDTLS_OPERATION_NONE if \p ctx has not been initialized. - */ -static inline mbedtls_operation_t mbedtls_cipher_get_operation( - const mbedtls_cipher_context_t *ctx ) -{ - MBEDTLS_INTERNAL_VALIDATE_RET( - ctx != NULL, MBEDTLS_OPERATION_NONE ); - if( ctx->MBEDTLS_PRIVATE(cipher_info) == NULL ) - return MBEDTLS_OPERATION_NONE; - - return ctx->MBEDTLS_PRIVATE(operation); -} - -/** - * \brief This function sets the key to use with the given context. - * - * \param ctx The generic cipher context. This must be initialized and - * bound to a cipher information structure. - * \param key The key to use. This must be a readable buffer of at - * least \p key_bitlen Bits. - * \param key_bitlen The key length to use, in Bits. - * \param operation The operation that the key will be used for: - * #MBEDTLS_ENCRYPT or #MBEDTLS_DECRYPT. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA on - * parameter-verification failure. - * \return A cipher-specific error code on failure. - */ -int mbedtls_cipher_setkey( mbedtls_cipher_context_t *ctx, - const unsigned char *key, - int key_bitlen, - const mbedtls_operation_t operation ); - -#if defined(MBEDTLS_CIPHER_MODE_WITH_PADDING) -/** - * \brief This function sets the padding mode, for cipher modes - * that use padding. - * - * The default passing mode is PKCS7 padding. - * - * \param ctx The generic cipher context. This must be initialized and - * bound to a cipher information structure. - * \param mode The padding mode. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CIPHER_FEATURE_UNAVAILABLE - * if the selected padding mode is not supported. - * \return #MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA if the cipher mode - * does not support padding. - */ -int mbedtls_cipher_set_padding_mode( mbedtls_cipher_context_t *ctx, - mbedtls_cipher_padding_t mode ); -#endif /* MBEDTLS_CIPHER_MODE_WITH_PADDING */ - -/** - * \brief This function sets the initialization vector (IV) - * or nonce. - * - * \note Some ciphers do not use IVs nor nonce. For these - * ciphers, this function has no effect. - * - * \param ctx The generic cipher context. This must be initialized and - * bound to a cipher information structure. - * \param iv The IV to use, or NONCE_COUNTER for CTR-mode ciphers. This - * must be a readable buffer of at least \p iv_len Bytes. - * \param iv_len The IV length for ciphers with variable-size IV. - * This parameter is discarded by ciphers with fixed-size IV. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA on - * parameter-verification failure. - */ -int mbedtls_cipher_set_iv( mbedtls_cipher_context_t *ctx, - const unsigned char *iv, - size_t iv_len ); - -/** - * \brief This function resets the cipher state. - * - * \note With non-AEAD ciphers, the order of calls for each message - * is as follows: - * 1. mbedtls_cipher_set_iv() if the mode uses an IV/nonce. - * 2. mbedtls_cipher_reset() - * 3. mbedtls_cipher_update() one or more times - * 4. mbedtls_cipher_finish() - * . - * This sequence can be repeated to encrypt or decrypt multiple - * messages with the same key. - * - * \note With AEAD ciphers, the order of calls for each message - * is as follows: - * 1. mbedtls_cipher_set_iv() if the mode uses an IV/nonce. - * 2. mbedtls_cipher_reset() - * 3. mbedtls_cipher_update_ad() - * 4. mbedtls_cipher_update() one or more times - * 5. mbedtls_cipher_finish() - * 6. mbedtls_cipher_check_tag() (for decryption) or - * mbedtls_cipher_write_tag() (for encryption). - * . - * This sequence can be repeated to encrypt or decrypt multiple - * messages with the same key. - * - * \param ctx The generic cipher context. This must be bound to a key. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA on - * parameter-verification failure. - */ -int mbedtls_cipher_reset( mbedtls_cipher_context_t *ctx ); - -#if defined(MBEDTLS_GCM_C) || defined(MBEDTLS_CHACHAPOLY_C) -/** - * \brief This function adds additional data for AEAD ciphers. - * Currently supported with GCM and ChaCha20+Poly1305. - * - * \param ctx The generic cipher context. This must be initialized. - * \param ad The additional data to use. This must be a readable - * buffer of at least \p ad_len Bytes. - * \param ad_len The length of \p ad in Bytes. - * - * \return \c 0 on success. - * \return A specific error code on failure. - */ -int mbedtls_cipher_update_ad( mbedtls_cipher_context_t *ctx, - const unsigned char *ad, size_t ad_len ); -#endif /* MBEDTLS_GCM_C || MBEDTLS_CHACHAPOLY_C */ - -/** - * \brief The generic cipher update function. It encrypts or - * decrypts using the given cipher context. Writes as - * many block-sized blocks of data as possible to output. - * Any data that cannot be written immediately is either - * added to the next block, or flushed when - * mbedtls_cipher_finish() is called. - * Exception: For MBEDTLS_MODE_ECB, expects a single block - * in size. For example, 16 Bytes for AES. - * - * \param ctx The generic cipher context. This must be initialized and - * bound to a key. - * \param input The buffer holding the input data. This must be a - * readable buffer of at least \p ilen Bytes. - * \param ilen The length of the input data. - * \param output The buffer for the output data. This must be able to - * hold at least `ilen + block_size`. This must not be the - * same buffer as \p input. - * \param olen The length of the output data, to be updated with the - * actual number of Bytes written. This must not be - * \c NULL. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA on - * parameter-verification failure. - * \return #MBEDTLS_ERR_CIPHER_FEATURE_UNAVAILABLE on an - * unsupported mode for a cipher. - * \return A cipher-specific error code on failure. - */ -int mbedtls_cipher_update( mbedtls_cipher_context_t *ctx, - const unsigned char *input, - size_t ilen, unsigned char *output, - size_t *olen ); - -/** - * \brief The generic cipher finalization function. If data still - * needs to be flushed from an incomplete block, the data - * contained in it is padded to the size of - * the last block, and written to the \p output buffer. - * - * \param ctx The generic cipher context. This must be initialized and - * bound to a key. - * \param output The buffer to write data to. This needs to be a writable - * buffer of at least \p block_size Bytes. - * \param olen The length of the data written to the \p output buffer. - * This may not be \c NULL. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA on - * parameter-verification failure. - * \return #MBEDTLS_ERR_CIPHER_FULL_BLOCK_EXPECTED on decryption - * expecting a full block but not receiving one. - * \return #MBEDTLS_ERR_CIPHER_INVALID_PADDING on invalid padding - * while decrypting. - * \return A cipher-specific error code on failure. - */ -int mbedtls_cipher_finish( mbedtls_cipher_context_t *ctx, - unsigned char *output, size_t *olen ); - -#if defined(MBEDTLS_GCM_C) || defined(MBEDTLS_CHACHAPOLY_C) -/** - * \brief This function writes a tag for AEAD ciphers. - * Currently supported with GCM and ChaCha20+Poly1305. - * This must be called after mbedtls_cipher_finish(). - * - * \param ctx The generic cipher context. This must be initialized, - * bound to a key, and have just completed a cipher - * operation through mbedtls_cipher_finish() the tag for - * which should be written. - * \param tag The buffer to write the tag to. This must be a writable - * buffer of at least \p tag_len Bytes. - * \param tag_len The length of the tag to write. - * - * \return \c 0 on success. - * \return A specific error code on failure. - */ -int mbedtls_cipher_write_tag( mbedtls_cipher_context_t *ctx, - unsigned char *tag, size_t tag_len ); - -/** - * \brief This function checks the tag for AEAD ciphers. - * Currently supported with GCM and ChaCha20+Poly1305. - * This must be called after mbedtls_cipher_finish(). - * - * \param ctx The generic cipher context. This must be initialized. - * \param tag The buffer holding the tag. This must be a readable - * buffer of at least \p tag_len Bytes. - * \param tag_len The length of the tag to check. - * - * \return \c 0 on success. - * \return A specific error code on failure. - */ -int mbedtls_cipher_check_tag( mbedtls_cipher_context_t *ctx, - const unsigned char *tag, size_t tag_len ); -#endif /* MBEDTLS_GCM_C || MBEDTLS_CHACHAPOLY_C */ - -/** - * \brief The generic all-in-one encryption/decryption function, - * for all ciphers except AEAD constructs. - * - * \param ctx The generic cipher context. This must be initialized. - * \param iv The IV to use, or NONCE_COUNTER for CTR-mode ciphers. - * This must be a readable buffer of at least \p iv_len - * Bytes. - * \param iv_len The IV length for ciphers with variable-size IV. - * This parameter is discarded by ciphers with fixed-size - * IV. - * \param input The buffer holding the input data. This must be a - * readable buffer of at least \p ilen Bytes. - * \param ilen The length of the input data in Bytes. - * \param output The buffer for the output data. This must be able to - * hold at least `ilen + block_size`. This must not be the - * same buffer as \p input. - * \param olen The length of the output data, to be updated with the - * actual number of Bytes written. This must not be - * \c NULL. - * - * \note Some ciphers do not use IVs nor nonce. For these - * ciphers, use \p iv = NULL and \p iv_len = 0. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA on - * parameter-verification failure. - * \return #MBEDTLS_ERR_CIPHER_FULL_BLOCK_EXPECTED on decryption - * expecting a full block but not receiving one. - * \return #MBEDTLS_ERR_CIPHER_INVALID_PADDING on invalid padding - * while decrypting. - * \return A cipher-specific error code on failure. - */ -int mbedtls_cipher_crypt( mbedtls_cipher_context_t *ctx, - const unsigned char *iv, size_t iv_len, - const unsigned char *input, size_t ilen, - unsigned char *output, size_t *olen ); - -#if defined(MBEDTLS_CIPHER_MODE_AEAD) || defined(MBEDTLS_NIST_KW_C) -/** - * \brief The authenticated encryption (AEAD/NIST_KW) function. - * - * \note For AEAD modes, the tag will be appended to the - * ciphertext, as recommended by RFC 5116. - * (NIST_KW doesn't have a separate tag.) - * - * \param ctx The generic cipher context. This must be initialized and - * bound to a key, with an AEAD algorithm or NIST_KW. - * \param iv The nonce to use. This must be a readable buffer of - * at least \p iv_len Bytes and may be \c NULL if \p - * iv_len is \c 0. - * \param iv_len The length of the nonce. For AEAD ciphers, this must - * satisfy the constraints imposed by the cipher used. - * For NIST_KW, this must be \c 0. - * \param ad The additional data to authenticate. This must be a - * readable buffer of at least \p ad_len Bytes, and may - * be \c NULL is \p ad_len is \c 0. - * \param ad_len The length of \p ad. For NIST_KW, this must be \c 0. - * \param input The buffer holding the input data. This must be a - * readable buffer of at least \p ilen Bytes, and may be - * \c NULL if \p ilen is \c 0. - * \param ilen The length of the input data. - * \param output The buffer for the output data. This must be a - * writable buffer of at least \p output_len Bytes, and - * must not be \c NULL. - * \param output_len The length of the \p output buffer in Bytes. For AEAD - * ciphers, this must be at least \p ilen + \p tag_len. - * For NIST_KW, this must be at least \p ilen + 8 - * (rounded up to a multiple of 8 if KWP is used); - * \p ilen + 15 is always a safe value. - * \param olen This will be filled with the actual number of Bytes - * written to the \p output buffer. This must point to a - * writable object of type \c size_t. - * \param tag_len The desired length of the authentication tag. For AEAD - * ciphers, this must match the constraints imposed by - * the cipher used, and in particular must not be \c 0. - * For NIST_KW, this must be \c 0. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA on - * parameter-verification failure. - * \return A cipher-specific error code on failure. - */ -int mbedtls_cipher_auth_encrypt_ext( mbedtls_cipher_context_t *ctx, - const unsigned char *iv, size_t iv_len, - const unsigned char *ad, size_t ad_len, - const unsigned char *input, size_t ilen, - unsigned char *output, size_t output_len, - size_t *olen, size_t tag_len ); - -/** - * \brief The authenticated encryption (AEAD/NIST_KW) function. - * - * \note If the data is not authentic, then the output buffer - * is zeroed out to prevent the unauthentic plaintext being - * used, making this interface safer. - * - * \note For AEAD modes, the tag must be appended to the - * ciphertext, as recommended by RFC 5116. - * (NIST_KW doesn't have a separate tag.) - * - * \param ctx The generic cipher context. This must be initialized and - * bound to a key, with an AEAD algorithm or NIST_KW. - * \param iv The nonce to use. This must be a readable buffer of - * at least \p iv_len Bytes and may be \c NULL if \p - * iv_len is \c 0. - * \param iv_len The length of the nonce. For AEAD ciphers, this must - * satisfy the constraints imposed by the cipher used. - * For NIST_KW, this must be \c 0. - * \param ad The additional data to authenticate. This must be a - * readable buffer of at least \p ad_len Bytes, and may - * be \c NULL is \p ad_len is \c 0. - * \param ad_len The length of \p ad. For NIST_KW, this must be \c 0. - * \param input The buffer holding the input data. This must be a - * readable buffer of at least \p ilen Bytes, and may be - * \c NULL if \p ilen is \c 0. - * \param ilen The length of the input data. For AEAD ciphers this - * must be at least \p tag_len. For NIST_KW this must be - * at least \c 8. - * \param output The buffer for the output data. This must be a - * writable buffer of at least \p output_len Bytes, and - * may be \c NULL if \p output_len is \c 0. - * \param output_len The length of the \p output buffer in Bytes. For AEAD - * ciphers, this must be at least \p ilen - \p tag_len. - * For NIST_KW, this must be at least \p ilen - 8. - * \param olen This will be filled with the actual number of Bytes - * written to the \p output buffer. This must point to a - * writable object of type \c size_t. - * \param tag_len The actual length of the authentication tag. For AEAD - * ciphers, this must match the constraints imposed by - * the cipher used, and in particular must not be \c 0. - * For NIST_KW, this must be \c 0. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA on - * parameter-verification failure. - * \return #MBEDTLS_ERR_CIPHER_AUTH_FAILED if data is not authentic. - * \return A cipher-specific error code on failure. - */ -int mbedtls_cipher_auth_decrypt_ext( mbedtls_cipher_context_t *ctx, - const unsigned char *iv, size_t iv_len, - const unsigned char *ad, size_t ad_len, - const unsigned char *input, size_t ilen, - unsigned char *output, size_t output_len, - size_t *olen, size_t tag_len ); -#endif /* MBEDTLS_CIPHER_MODE_AEAD || MBEDTLS_NIST_KW_C */ -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_CIPHER_H */ diff --git a/windows/mbedtls/cmac.h b/windows/mbedtls/cmac.h deleted file mode 100644 index 3e599b9d..00000000 --- a/windows/mbedtls/cmac.h +++ /dev/null @@ -1,223 +0,0 @@ -/** - * \file cmac.h - * - * \brief This file contains CMAC definitions and functions. - * - * The Cipher-based Message Authentication Code (CMAC) Mode for - * Authentication is defined in RFC-4493: The AES-CMAC Algorithm. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_CMAC_H -#define MBEDTLS_CMAC_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/cipher.h" - -#ifdef __cplusplus -extern "C" { -#endif - -#define MBEDTLS_AES_BLOCK_SIZE 16 -#define MBEDTLS_DES3_BLOCK_SIZE 8 - -#if defined(MBEDTLS_AES_C) -#define MBEDTLS_CIPHER_BLKSIZE_MAX 16 /**< The longest block used by CMAC is that of AES. */ -#else -#define MBEDTLS_CIPHER_BLKSIZE_MAX 8 /**< The longest block used by CMAC is that of 3DES. */ -#endif - -#if !defined(MBEDTLS_CMAC_ALT) - -/** - * The CMAC context structure. - */ -struct mbedtls_cmac_context_t -{ - /** The internal state of the CMAC algorithm. */ - unsigned char MBEDTLS_PRIVATE(state)[MBEDTLS_CIPHER_BLKSIZE_MAX]; - - /** Unprocessed data - either data that was not block aligned and is still - * pending processing, or the final block. */ - unsigned char MBEDTLS_PRIVATE(unprocessed_block)[MBEDTLS_CIPHER_BLKSIZE_MAX]; - - /** The length of data pending processing. */ - size_t MBEDTLS_PRIVATE(unprocessed_len); -}; - -#else /* !MBEDTLS_CMAC_ALT */ -#include "cmac_alt.h" -#endif /* !MBEDTLS_CMAC_ALT */ - -/** - * \brief This function sets the CMAC key, and prepares to authenticate - * the input data. - * Must be called with an initialized cipher context. - * - * \note When the CMAC implementation is supplied by an alternate - * implementation (through #MBEDTLS_CMAC_ALT), some ciphers - * may not be supported by that implementation, and thus - * return an error. Alternate implementations must support - * AES-128 and AES-256, and may support AES-192 and 3DES. - * - * \param ctx The cipher context used for the CMAC operation, initialized - * as one of the following types: MBEDTLS_CIPHER_AES_128_ECB, - * MBEDTLS_CIPHER_AES_192_ECB, MBEDTLS_CIPHER_AES_256_ECB, - * or MBEDTLS_CIPHER_DES_EDE3_ECB. - * \param key The CMAC key. - * \param keybits The length of the CMAC key in bits. - * Must be supported by the cipher. - * - * \return \c 0 on success. - * \return A cipher-specific error code on failure. - */ -int mbedtls_cipher_cmac_starts( mbedtls_cipher_context_t *ctx, - const unsigned char *key, size_t keybits ); - -/** - * \brief This function feeds an input buffer into an ongoing CMAC - * computation. - * - * It is called between mbedtls_cipher_cmac_starts() or - * mbedtls_cipher_cmac_reset(), and mbedtls_cipher_cmac_finish(). - * Can be called repeatedly. - * - * \param ctx The cipher context used for the CMAC operation. - * \param input The buffer holding the input data. - * \param ilen The length of the input data. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA - * if parameter verification fails. - */ -int mbedtls_cipher_cmac_update( mbedtls_cipher_context_t *ctx, - const unsigned char *input, size_t ilen ); - -/** - * \brief This function finishes the CMAC operation, and writes - * the result to the output buffer. - * - * It is called after mbedtls_cipher_cmac_update(). - * It can be followed by mbedtls_cipher_cmac_reset() and - * mbedtls_cipher_cmac_update(), or mbedtls_cipher_free(). - * - * \param ctx The cipher context used for the CMAC operation. - * \param output The output buffer for the CMAC checksum result. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA - * if parameter verification fails. - */ -int mbedtls_cipher_cmac_finish( mbedtls_cipher_context_t *ctx, - unsigned char *output ); - -/** - * \brief This function prepares the authentication of another - * message with the same key as the previous CMAC - * operation. - * - * It is called after mbedtls_cipher_cmac_finish() - * and before mbedtls_cipher_cmac_update(). - * - * \param ctx The cipher context used for the CMAC operation. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA - * if parameter verification fails. - */ -int mbedtls_cipher_cmac_reset( mbedtls_cipher_context_t *ctx ); - -/** - * \brief This function calculates the full generic CMAC - * on the input buffer with the provided key. - * - * The function allocates the context, performs the - * calculation, and frees the context. - * - * The CMAC result is calculated as - * output = generic CMAC(cmac key, input buffer). - * - * \note When the CMAC implementation is supplied by an alternate - * implementation (through #MBEDTLS_CMAC_ALT), some ciphers - * may not be supported by that implementation, and thus - * return an error. Alternate implementations must support - * AES-128 and AES-256, and may support AES-192 and 3DES. - * - * \param cipher_info The cipher information. - * \param key The CMAC key. - * \param keylen The length of the CMAC key in bits. - * \param input The buffer holding the input data. - * \param ilen The length of the input data. - * \param output The buffer for the generic CMAC result. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA - * if parameter verification fails. - */ -int mbedtls_cipher_cmac( const mbedtls_cipher_info_t *cipher_info, - const unsigned char *key, size_t keylen, - const unsigned char *input, size_t ilen, - unsigned char *output ); - -#if defined(MBEDTLS_AES_C) -/** - * \brief This function implements the AES-CMAC-PRF-128 pseudorandom - * function, as defined in - * RFC-4615: The Advanced Encryption Standard-Cipher-based - * Message Authentication Code-Pseudo-Random Function-128 - * (AES-CMAC-PRF-128) Algorithm for the Internet Key - * Exchange Protocol (IKE). - * - * \param key The key to use. - * \param key_len The key length in Bytes. - * \param input The buffer holding the input data. - * \param in_len The length of the input data in Bytes. - * \param output The buffer holding the generated 16 Bytes of - * pseudorandom output. - * - * \return \c 0 on success. - */ -int mbedtls_aes_cmac_prf_128( const unsigned char *key, size_t key_len, - const unsigned char *input, size_t in_len, - unsigned char output[16] ); -#endif /* MBEDTLS_AES_C */ - -#if defined(MBEDTLS_SELF_TEST) && ( defined(MBEDTLS_AES_C) || defined(MBEDTLS_DES_C) ) -/** - * \brief The CMAC checkup routine. - * - * \note In case the CMAC routines are provided by an alternative - * implementation (i.e. #MBEDTLS_CMAC_ALT is defined), the - * checkup routine will succeed even if the implementation does - * not support the less widely used AES-192 or 3DES primitives. - * The self-test requires at least AES-128 and AES-256 to be - * supported by the underlying implementation. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_cmac_self_test( int verbose ); -#endif /* MBEDTLS_SELF_TEST && ( MBEDTLS_AES_C || MBEDTLS_DES_C ) */ - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_CMAC_H */ diff --git a/windows/mbedtls/compat-2.x.h b/windows/mbedtls/compat-2.x.h deleted file mode 100644 index cdf81dcb..00000000 --- a/windows/mbedtls/compat-2.x.h +++ /dev/null @@ -1,58 +0,0 @@ -/** - * \file compat-2.x.h - * - * \brief Compatibility definitions - * - * \deprecated Use the new names directly instead - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#if defined(MBEDTLS_DEPRECATED_WARNING) -#warning "Including compat-2.x.h is deprecated" -#endif - -#ifndef MBEDTLS_COMPAT2X_H -#define MBEDTLS_COMPAT2X_H - -/* - * Macros for renamed functions - */ -#define mbedtls_ctr_drbg_update_ret mbedtls_ctr_drbg_update -#define mbedtls_hmac_drbg_update_ret mbedtls_hmac_drbg_update -#define mbedtls_md5_starts_ret mbedtls_md5_starts -#define mbedtls_md5_update_ret mbedtls_md5_update -#define mbedtls_md5_finish_ret mbedtls_md5_finish -#define mbedtls_md5_ret mbedtls_md5 -#define mbedtls_ripemd160_starts_ret mbedtls_ripemd160_starts -#define mbedtls_ripemd160_update_ret mbedtls_ripemd160_update -#define mbedtls_ripemd160_finish_ret mbedtls_ripemd160_finish -#define mbedtls_ripemd160_ret mbedtls_ripemd160 -#define mbedtls_sha1_starts_ret mbedtls_sha1_starts -#define mbedtls_sha1_update_ret mbedtls_sha1_update -#define mbedtls_sha1_finish_ret mbedtls_sha1_finish -#define mbedtls_sha1_ret mbedtls_sha1 -#define mbedtls_sha256_starts_ret mbedtls_sha256_starts -#define mbedtls_sha256_update_ret mbedtls_sha256_update -#define mbedtls_sha256_finish_ret mbedtls_sha256_finish -#define mbedtls_sha256_ret mbedtls_sha256 -#define mbedtls_sha512_starts_ret mbedtls_sha512_starts -#define mbedtls_sha512_update_ret mbedtls_sha512_update -#define mbedtls_sha512_finish_ret mbedtls_sha512_finish -#define mbedtls_sha512_ret mbedtls_sha512 - -#endif /* MBEDTLS_COMPAT2X_H */ diff --git a/windows/mbedtls/config_psa.h b/windows/mbedtls/config_psa.h deleted file mode 100644 index f182c527..00000000 --- a/windows/mbedtls/config_psa.h +++ /dev/null @@ -1,770 +0,0 @@ -/** - * \file mbedtls/config_psa.h - * \brief PSA crypto configuration options (set of defines) - * - * This set of compile-time options takes settings defined in - * include/mbedtls/mbedtls_config.h and include/psa/crypto_config.h and uses - * those definitions to define symbols used in the library code. - * - * Users and integrators should not edit this file, please edit - * include/mbedtls/mbedtls_config.h for MBETLS_XXX settings or - * include/psa/crypto_config.h for PSA_WANT_XXX settings. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_CONFIG_PSA_H -#define MBEDTLS_CONFIG_PSA_H - -#if defined(MBEDTLS_PSA_CRYPTO_CONFIG) -#include "psa/crypto_config.h" -#endif /* defined(MBEDTLS_PSA_CRYPTO_CONFIG) */ - -#ifdef __cplusplus -extern "C" { -#endif - - - -/****************************************************************/ -/* De facto synonyms */ -/****************************************************************/ - -#if defined(PSA_WANT_ALG_ECDSA_ANY) && !defined(PSA_WANT_ALG_ECDSA) -#define PSA_WANT_ALG_ECDSA PSA_WANT_ALG_ECDSA_ANY -#elif !defined(PSA_WANT_ALG_ECDSA_ANY) && defined(PSA_WANT_ALG_ECDSA) -#define PSA_WANT_ALG_ECDSA_ANY PSA_WANT_ALG_ECDSA -#endif - -#if defined(PSA_WANT_ALG_RSA_PKCS1V15_SIGN_RAW) && !defined(PSA_WANT_ALG_RSA_PKCS1V15_SIGN) -#define PSA_WANT_ALG_RSA_PKCS1V15_SIGN PSA_WANT_ALG_RSA_PKCS1V15_SIGN_RAW -#elif !defined(PSA_WANT_ALG_RSA_PKCS1V15_SIGN_RAW) && defined(PSA_WANT_ALG_RSA_PKCS1V15_SIGN) -#define PSA_WANT_ALG_RSA_PKCS1V15_SIGN_RAW PSA_WANT_ALG_RSA_PKCS1V15_SIGN -#endif - - - -/****************************************************************/ -/* Require built-in implementations based on PSA requirements */ -/****************************************************************/ - -#if defined(MBEDTLS_PSA_CRYPTO_CONFIG) - -#if defined(PSA_WANT_ALG_DETERMINISTIC_ECDSA) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_DETERMINISTIC_ECDSA) -#define MBEDTLS_PSA_BUILTIN_ALG_DETERMINISTIC_ECDSA 1 -#define MBEDTLS_ECDSA_DETERMINISTIC -#define MBEDTLS_ECDSA_C -#define MBEDTLS_HMAC_DRBG_C -#define MBEDTLS_MD_C -#endif /* !MBEDTLS_PSA_ACCEL_ALG_DETERMINISTIC_ECDSA */ -#endif /* PSA_WANT_ALG_DETERMINISTIC_ECDSA */ - -#if defined(PSA_WANT_ALG_ECDH) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_ECDH) -#define MBEDTLS_PSA_BUILTIN_ALG_ECDH 1 -#define MBEDTLS_ECDH_C -#define MBEDTLS_ECP_C -#define MBEDTLS_BIGNUM_C -#endif /* !MBEDTLS_PSA_ACCEL_ALG_ECDH */ -#endif /* PSA_WANT_ALG_ECDH */ - -#if defined(PSA_WANT_ALG_ECDSA) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_ECDSA) -#define MBEDTLS_PSA_BUILTIN_ALG_ECDSA 1 -#define MBEDTLS_ECDSA_C -#endif /* !MBEDTLS_PSA_ACCEL_ALG_ECDSA */ -#endif /* PSA_WANT_ALG_ECDSA */ - -#if defined(PSA_WANT_ALG_HKDF) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_HKDF) -#define MBEDTLS_PSA_BUILTIN_ALG_HMAC 1 -#define MBEDTLS_PSA_BUILTIN_ALG_HKDF 1 -#endif /* !MBEDTLS_PSA_ACCEL_ALG_HKDF */ -#endif /* PSA_WANT_ALG_HKDF */ - -#if defined(PSA_WANT_ALG_HMAC) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_HMAC) -#define MBEDTLS_PSA_BUILTIN_ALG_HMAC 1 -#endif /* !MBEDTLS_PSA_ACCEL_ALG_HMAC */ -#endif /* PSA_WANT_ALG_HMAC */ - -#if defined(PSA_WANT_ALG_MD5) && !defined(MBEDTLS_PSA_ACCEL_ALG_MD5) -#define MBEDTLS_PSA_BUILTIN_ALG_MD5 1 -#define MBEDTLS_MD5_C -#endif - -#if defined(PSA_WANT_ALG_RIPEMD160) && !defined(MBEDTLS_PSA_ACCEL_ALG_RIPEMD160) -#define MBEDTLS_PSA_BUILTIN_ALG_RIPEMD160 1 -#define MBEDTLS_RIPEMD160_C -#endif - -#if defined(PSA_WANT_ALG_RSA_OAEP) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_RSA_OAEP) -#define MBEDTLS_PSA_BUILTIN_ALG_RSA_OAEP 1 -#define MBEDTLS_RSA_C -#define MBEDTLS_BIGNUM_C -#define MBEDTLS_OID_C -#define MBEDTLS_PKCS1_V21 -#define MBEDTLS_MD_C -#endif /* !MBEDTLS_PSA_ACCEL_ALG_RSA_OAEP */ -#endif /* PSA_WANT_ALG_RSA_OAEP */ - -#if defined(PSA_WANT_ALG_RSA_PKCS1V15_CRYPT) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_RSA_PKCS1V15_CRYPT) -#define MBEDTLS_PSA_BUILTIN_ALG_RSA_PKCS1V15_CRYPT 1 -#define MBEDTLS_RSA_C -#define MBEDTLS_BIGNUM_C -#define MBEDTLS_OID_C -#define MBEDTLS_PKCS1_V15 -#endif /* !MBEDTLS_PSA_ACCEL_ALG_RSA_PKCS1V15_CRYPT */ -#endif /* PSA_WANT_ALG_RSA_PKCS1V15_CRYPT */ - -#if defined(PSA_WANT_ALG_RSA_PKCS1V15_SIGN) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_RSA_PKCS1V15_SIGN) -#define MBEDTLS_PSA_BUILTIN_ALG_RSA_PKCS1V15_SIGN 1 -#define MBEDTLS_RSA_C -#define MBEDTLS_BIGNUM_C -#define MBEDTLS_OID_C -#define MBEDTLS_PKCS1_V15 -#define MBEDTLS_MD_C -#endif /* !MBEDTLS_PSA_ACCEL_ALG_RSA_PKCS1V15_SIGN */ -#endif /* PSA_WANT_ALG_RSA_PKCS1V15_SIGN */ - -#if defined(PSA_WANT_ALG_RSA_PSS) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_RSA_PSS) -#define MBEDTLS_PSA_BUILTIN_ALG_RSA_PSS 1 -#define MBEDTLS_RSA_C -#define MBEDTLS_BIGNUM_C -#define MBEDTLS_OID_C -#define MBEDTLS_PKCS1_V21 -#define MBEDTLS_MD_C -#endif /* !MBEDTLS_PSA_ACCEL_ALG_RSA_PSS */ -#endif /* PSA_WANT_ALG_RSA_PSS */ - -#if defined(PSA_WANT_ALG_SHA_1) && !defined(MBEDTLS_PSA_ACCEL_ALG_SHA_1) -#define MBEDTLS_PSA_BUILTIN_ALG_SHA_1 1 -#define MBEDTLS_SHA1_C -#endif - -#if defined(PSA_WANT_ALG_SHA_224) && !defined(MBEDTLS_PSA_ACCEL_ALG_SHA_224) -#define MBEDTLS_PSA_BUILTIN_ALG_SHA_224 1 -#define MBEDTLS_SHA224_C -#endif - -#if defined(PSA_WANT_ALG_SHA_256) && !defined(MBEDTLS_PSA_ACCEL_ALG_SHA_256) -#define MBEDTLS_PSA_BUILTIN_ALG_SHA_256 1 -#define MBEDTLS_SHA256_C -#endif - -#if defined(PSA_WANT_ALG_SHA_384) && !defined(MBEDTLS_PSA_ACCEL_ALG_SHA_384) -#define MBEDTLS_PSA_BUILTIN_ALG_SHA_384 1 -#define MBEDTLS_SHA384_C -#endif - -#if defined(PSA_WANT_ALG_SHA_512) && !defined(MBEDTLS_PSA_ACCEL_ALG_SHA_512) -#define MBEDTLS_PSA_BUILTIN_ALG_SHA_512 1 -#define MBEDTLS_SHA512_C -#endif - -#if defined(PSA_WANT_ALG_TLS12_PRF) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_TLS12_PRF) -#define MBEDTLS_PSA_BUILTIN_ALG_TLS12_PRF 1 -#endif /* !MBEDTLS_PSA_ACCEL_ALG_TLS12_PRF */ -#endif /* PSA_WANT_ALG_TLS12_PRF */ - -#if defined(PSA_WANT_ALG_TLS12_PSK_TO_MS) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_TLS12_PSK_TO_MS) -#define MBEDTLS_PSA_BUILTIN_ALG_TLS12_PSK_TO_MS 1 -#endif /* !MBEDTLS_PSA_ACCEL_ALG_TLS12_PSK_TO_MS */ -#endif /* PSA_WANT_ALG_TLS12_PSK_TO_MS */ - -#if defined(PSA_WANT_KEY_TYPE_ECC_KEY_PAIR) -#if !defined(MBEDTLS_PSA_ACCEL_KEY_TYPE_ECC_KEY_PAIR) -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_ECC_KEY_PAIR 1 -#define MBEDTLS_ECP_C -#define MBEDTLS_BIGNUM_C -#endif /* !MBEDTLS_PSA_ACCEL_KEY_TYPE_ECC_KEY_PAIR */ -#endif /* PSA_WANT_KEY_TYPE_ECC_KEY_PAIR */ - -#if defined(PSA_WANT_KEY_TYPE_ECC_PUBLIC_KEY) -#if !defined(MBEDTLS_PSA_ACCEL_KEY_TYPE_ECC_PUBLIC_KEY) -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_ECC_PUBLIC_KEY 1 -#define MBEDTLS_ECP_C -#define MBEDTLS_BIGNUM_C -#endif /* !MBEDTLS_PSA_ACCEL_KEY_TYPE_ECC_PUBLIC_KEY */ -#endif /* PSA_WANT_KEY_TYPE_ECC_PUBLIC_KEY */ - -#if defined(PSA_WANT_KEY_TYPE_RSA_KEY_PAIR) -#if !defined(MBEDTLS_PSA_ACCEL_KEY_TYPE_RSA_KEY_PAIR) -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_RSA_KEY_PAIR 1 -#define MBEDTLS_RSA_C -#define MBEDTLS_BIGNUM_C -#define MBEDTLS_OID_C -#define MBEDTLS_GENPRIME -#define MBEDTLS_PK_PARSE_C -#define MBEDTLS_PK_WRITE_C -#define MBEDTLS_PK_C -#endif /* !MBEDTLS_PSA_ACCEL_KEY_TYPE_RSA_KEY_PAIR */ -#endif /* PSA_WANT_KEY_TYPE_RSA_KEY_PAIR */ - -#if defined(PSA_WANT_KEY_TYPE_RSA_PUBLIC_KEY) -#if !defined(MBEDTLS_PSA_ACCEL_KEY_TYPE_RSA_PUBLIC_KEY) -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_RSA_PUBLIC_KEY 1 -#define MBEDTLS_RSA_C -#define MBEDTLS_BIGNUM_C -#define MBEDTLS_OID_C -#define MBEDTLS_PK_PARSE_C -#define MBEDTLS_PK_WRITE_C -#define MBEDTLS_PK_C -#endif /* !MBEDTLS_PSA_ACCEL_KEY_TYPE_RSA_PUBLIC_KEY */ -#endif /* PSA_WANT_KEY_TYPE_RSA_PUBLIC_KEY */ - -/* If any of the block modes are requested that don't have an - * associated HW assist, define PSA_HAVE_SOFT_BLOCK_MODE for checking - * in the block cipher key types. */ -#if (defined(PSA_WANT_ALG_CTR) && !defined(MBEDTLS_PSA_ACCEL_ALG_CTR)) || \ - (defined(PSA_WANT_ALG_CFB) && !defined(MBEDTLS_PSA_ACCEL_ALG_CFB)) || \ - (defined(PSA_WANT_ALG_OFB) && !defined(MBEDTLS_PSA_ACCEL_ALG_OFB)) || \ - (defined(PSA_WANT_ALG_XTS) && !defined(MBEDTLS_PSA_ACCEL_ALG_XTS)) || \ - defined(PSA_WANT_ALG_ECB_NO_PADDING) || \ - (defined(PSA_WANT_ALG_CBC_NO_PADDING) && \ - !defined(MBEDTLS_PSA_ACCEL_ALG_CBC_NO_PADDING)) || \ - (defined(PSA_WANT_ALG_CBC_PKCS7) && \ - !defined(MBEDTLS_PSA_ACCEL_ALG_CBC_PKCS7)) || \ - (defined(PSA_WANT_ALG_CMAC) && !defined(MBEDTLS_PSA_ACCEL_ALG_CMAC)) -#define PSA_HAVE_SOFT_BLOCK_MODE 1 -#endif - -#if (defined(PSA_WANT_ALG_GCM) && !defined(MBEDTLS_PSA_ACCEL_ALG_GCM)) || \ - (defined(PSA_WANT_ALG_CCM) && !defined(MBEDTLS_PSA_ACCEL_ALG_CCM)) -#define PSA_HAVE_SOFT_BLOCK_AEAD 1 -#endif - -#if defined(PSA_WANT_KEY_TYPE_AES) -#if !defined(MBEDTLS_PSA_ACCEL_KEY_TYPE_AES) -#define PSA_HAVE_SOFT_KEY_TYPE_AES 1 -#endif /* !MBEDTLS_PSA_ACCEL_KEY_TYPE_AES */ -#if defined(PSA_HAVE_SOFT_KEY_TYPE_AES) || \ - defined(PSA_HAVE_SOFT_BLOCK_MODE) || \ - defined(PSA_HAVE_SOFT_BLOCK_AEAD) -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_AES 1 -#define MBEDTLS_AES_C -#endif /* PSA_HAVE_SOFT_KEY_TYPE_AES || PSA_HAVE_SOFT_BLOCK_MODE */ -#endif /* PSA_WANT_KEY_TYPE_AES */ - -#if defined(PSA_WANT_KEY_TYPE_CAMELLIA) -#if !defined(MBEDTLS_PSA_ACCEL_KEY_TYPE_CAMELLIA) -#define PSA_HAVE_SOFT_KEY_TYPE_CAMELLIA 1 -#endif /* !MBEDTLS_PSA_ACCEL_KEY_TYPE_CAMELLIA */ -#if defined(PSA_HAVE_SOFT_KEY_TYPE_CAMELLIA) || \ - defined(PSA_HAVE_SOFT_BLOCK_MODE) || \ - defined(PSA_HAVE_SOFT_BLOCK_AEAD) -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_CAMELLIA 1 -#define MBEDTLS_CAMELLIA_C -#endif /* PSA_HAVE_SOFT_KEY_TYPE_CAMELLIA || PSA_HAVE_SOFT_BLOCK_MODE */ -#endif /* PSA_WANT_KEY_TYPE_CAMELLIA */ - -#if defined(PSA_WANT_KEY_TYPE_DES) -#if !defined(MBEDTLS_PSA_ACCEL_KEY_TYPE_DES) -#define PSA_HAVE_SOFT_KEY_TYPE_DES 1 -#endif /* !MBEDTLS_PSA_ACCEL_KEY_TYPE_DES */ -#if defined(PSA_HAVE_SOFT_KEY_TYPE_DES) || \ - defined(PSA_HAVE_SOFT_BLOCK_MODE) -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_DES 1 -#define MBEDTLS_DES_C -#endif /*PSA_HAVE_SOFT_KEY_TYPE_DES || PSA_HAVE_SOFT_BLOCK_MODE */ -#endif /* PSA_WANT_KEY_TYPE_DES */ - -#if defined(PSA_WANT_KEY_TYPE_CHACHA20) -#if !defined(MBEDTLS_PSA_ACCEL_KEY_TYPE_CHACHA20) -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_CHACHA20 1 -#define MBEDTLS_CHACHA20_C -#endif /*!MBEDTLS_PSA_ACCEL_KEY_TYPE_CHACHA20 */ -#endif /* PSA_WANT_KEY_TYPE_CHACHA20 */ - -/* If any of the software block ciphers are selected, define - * PSA_HAVE_SOFT_BLOCK_CIPHER, which can be used in any of these - * situations. */ -#if defined(PSA_HAVE_SOFT_KEY_TYPE_AES) || \ - defined(PSA_HAVE_SOFT_KEY_TYPE_DES) || \ - defined(PSA_HAVE_SOFT_KEY_TYPE_CAMELLIA) -#define PSA_HAVE_SOFT_BLOCK_CIPHER 1 -#endif - -#if defined(PSA_WANT_ALG_STREAM_CIPHER) -#define MBEDTLS_PSA_BUILTIN_ALG_STREAM_CIPHER 1 -#endif /* PSA_WANT_ALG_STREAM_CIPHER */ - -#if defined(PSA_WANT_ALG_CBC_MAC) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_CBC_MAC) -#error "CBC-MAC is not yet supported via the PSA API in Mbed TLS." -#define MBEDTLS_PSA_BUILTIN_ALG_CBC_MAC 1 -#endif /* !MBEDTLS_PSA_ACCEL_ALG_CBC_MAC */ -#endif /* PSA_WANT_ALG_CBC_MAC */ - -#if defined(PSA_WANT_ALG_CMAC) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_CMAC) || \ - defined(PSA_HAVE_SOFT_BLOCK_CIPHER) -#define MBEDTLS_PSA_BUILTIN_ALG_CMAC 1 -#define MBEDTLS_CMAC_C -#endif /* !MBEDTLS_PSA_ACCEL_ALG_CMAC */ -#endif /* PSA_WANT_ALG_CMAC */ - -#if defined(PSA_WANT_ALG_CTR) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_CTR) || \ - defined(PSA_HAVE_SOFT_BLOCK_CIPHER) -#define MBEDTLS_PSA_BUILTIN_ALG_CTR 1 -#define MBEDTLS_CIPHER_MODE_CTR -#endif -#endif /* PSA_WANT_ALG_CTR */ - -#if defined(PSA_WANT_ALG_CFB) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_CFB) || \ - defined(PSA_HAVE_SOFT_BLOCK_CIPHER) -#define MBEDTLS_PSA_BUILTIN_ALG_CFB 1 -#define MBEDTLS_CIPHER_MODE_CFB -#endif -#endif /* PSA_WANT_ALG_CFB */ - -#if defined(PSA_WANT_ALG_OFB) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_OFB) || \ - defined(PSA_HAVE_SOFT_BLOCK_CIPHER) -#define MBEDTLS_PSA_BUILTIN_ALG_OFB 1 -#define MBEDTLS_CIPHER_MODE_OFB -#endif -#endif /* PSA_WANT_ALG_OFB */ - -#if defined(PSA_WANT_ALG_XTS) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_XTS) || \ - defined(PSA_HAVE_SOFT_BLOCK_CIPHER) -#define MBEDTLS_PSA_BUILTIN_ALG_XTS 1 -#define MBEDTLS_CIPHER_MODE_XTS -#endif -#endif /* PSA_WANT_ALG_XTS */ - -#if defined(PSA_WANT_ALG_ECB_NO_PADDING) -#define MBEDTLS_PSA_BUILTIN_ALG_ECB_NO_PADDING 1 -#endif - -#if defined(PSA_WANT_ALG_CBC_NO_PADDING) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_CBC_NO_PADDING) || \ - defined(PSA_HAVE_SOFT_BLOCK_CIPHER) -#define MBEDTLS_CIPHER_MODE_CBC -#define MBEDTLS_PSA_BUILTIN_ALG_CBC_NO_PADDING 1 -#endif -#endif /* PSA_WANT_ALG_CBC_NO_PADDING */ - -#if defined(PSA_WANT_ALG_CBC_PKCS7) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_CBC_PKCS7) || \ - defined(PSA_HAVE_SOFT_BLOCK_CIPHER) -#define MBEDTLS_CIPHER_MODE_CBC -#define MBEDTLS_PSA_BUILTIN_ALG_CBC_PKCS7 1 -#define MBEDTLS_CIPHER_PADDING_PKCS7 -#endif -#endif /* PSA_WANT_ALG_CBC_PKCS7 */ - -#if defined(PSA_WANT_ALG_CCM) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_CCM) || \ - defined(PSA_HAVE_SOFT_KEY_TYPE_AES) || \ - defined(PSA_HAVE_SOFT_KEY_TYPE_CAMELLIA) -#define MBEDTLS_PSA_BUILTIN_ALG_CCM 1 -#define MBEDTLS_CCM_C -#endif -#endif /* PSA_WANT_ALG_CCM */ - -#if defined(PSA_WANT_ALG_GCM) -#if !defined(MBEDTLS_PSA_ACCEL_ALG_GCM) || \ - defined(PSA_HAVE_SOFT_KEY_TYPE_AES) || \ - defined(PSA_HAVE_SOFT_KEY_TYPE_CAMELLIA) -#define MBEDTLS_PSA_BUILTIN_ALG_GCM 1 -#define MBEDTLS_GCM_C -#endif -#endif /* PSA_WANT_ALG_GCM */ - -#if defined(PSA_WANT_ALG_CHACHA20_POLY1305) -#if defined(PSA_WANT_KEY_TYPE_CHACHA20) -#define MBEDTLS_CHACHAPOLY_C -#define MBEDTLS_PSA_BUILTIN_ALG_CHACHA20_POLY1305 1 -#endif /* PSA_WANT_KEY_TYPE_CHACHA20 */ -#endif /* PSA_WANT_ALG_CHACHA20_POLY1305 */ - -#if defined(PSA_WANT_ECC_BRAINPOOL_P_R1_256) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_BRAINPOOL_P_R1_256) -#define MBEDTLS_ECP_DP_BP256R1_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_BRAINPOOL_P_R1_256 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_BRAINPOOL_P_R1_256 */ -#endif /* PSA_WANT_ECC_BRAINPOOL_P_R1_256 */ - -#if defined(PSA_WANT_ECC_BRAINPOOL_P_R1_384) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_BRAINPOOL_P_R1_384) -#define MBEDTLS_ECP_DP_BP384R1_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_BRAINPOOL_P_R1_384 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_BRAINPOOL_P_R1_384 */ -#endif /* PSA_WANT_ECC_BRAINPOOL_P_R1_384 */ - -#if defined(PSA_WANT_ECC_BRAINPOOL_P_R1_512) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_BRAINPOOL_P_R1_512) -#define MBEDTLS_ECP_DP_BP512R1_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_BRAINPOOL_P_R1_512 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_BRAINPOOL_P_R1_512 */ -#endif /* PSA_WANT_ECC_BRAINPOOL_P_R1_512 */ - -#if defined(PSA_WANT_ECC_MONTGOMERY_255) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_MONTGOMERY_255) -#define MBEDTLS_ECP_DP_CURVE25519_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_MONTGOMERY_255 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_MONTGOMERY_255 */ -#endif /* PSA_WANT_ECC_MONTGOMERY_255 */ - -#if defined(PSA_WANT_ECC_MONTGOMERY_448) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_MONTGOMERY_448) -/* - * Curve448 is not yet supported via the PSA API in Mbed TLS - * (https://github.com/ARMmbed/mbedtls/issues/4249). - */ -#error "Curve448 is not yet supported via the PSA API in Mbed TLS." -#define MBEDTLS_ECP_DP_CURVE448_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_MONTGOMERY_448 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_MONTGOMERY_448 */ -#endif /* PSA_WANT_ECC_MONTGOMERY_448 */ - -#if defined(PSA_WANT_ECC_SECP_R1_192) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_SECP_R1_192) -#define MBEDTLS_ECP_DP_SECP192R1_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_R1_192 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_SECP_R1_192 */ -#endif /* PSA_WANT_ECC_SECP_R1_192 */ - -#if defined(PSA_WANT_ECC_SECP_R1_224) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_SECP_R1_224) -#define MBEDTLS_ECP_DP_SECP224R1_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_R1_224 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_SECP_R1_224 */ -#endif /* PSA_WANT_ECC_SECP_R1_224 */ - -#if defined(PSA_WANT_ECC_SECP_R1_256) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_SECP_R1_256) -#define MBEDTLS_ECP_DP_SECP256R1_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_R1_256 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_SECP_R1_256 */ -#endif /* PSA_WANT_ECC_SECP_R1_256 */ - -#if defined(PSA_WANT_ECC_SECP_R1_384) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_SECP_R1_384) -#define MBEDTLS_ECP_DP_SECP384R1_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_R1_384 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_SECP_R1_384 */ -#endif /* PSA_WANT_ECC_SECP_R1_384 */ - -#if defined(PSA_WANT_ECC_SECP_R1_521) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_SECP_R1_521) -#define MBEDTLS_ECP_DP_SECP521R1_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_R1_521 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_SECP_R1_521 */ -#endif /* PSA_WANT_ECC_SECP_R1_521 */ - -#if defined(PSA_WANT_ECC_SECP_K1_192) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_SECP_K1_192) -#define MBEDTLS_ECP_DP_SECP192K1_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_K1_192 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_SECP_K1_192 */ -#endif /* PSA_WANT_ECC_SECP_K1_192 */ - -#if defined(PSA_WANT_ECC_SECP_K1_224) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_SECP_K1_224) -/* - * SECP224K1 is buggy via the PSA API in Mbed TLS - * (https://github.com/ARMmbed/mbedtls/issues/3541). - */ -#error "SECP224K1 is buggy via the PSA API in Mbed TLS." -#define MBEDTLS_ECP_DP_SECP224K1_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_K1_224 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_SECP_K1_224 */ -#endif /* PSA_WANT_ECC_SECP_K1_224 */ - -#if defined(PSA_WANT_ECC_SECP_K1_256) -#if !defined(MBEDTLS_PSA_ACCEL_ECC_SECP_K1_256) -#define MBEDTLS_ECP_DP_SECP256K1_ENABLED -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_K1_256 1 -#endif /* !MBEDTLS_PSA_ACCEL_ECC_SECP_K1_256 */ -#endif /* PSA_WANT_ECC_SECP_K1_256 */ - - - -/****************************************************************/ -/* Infer PSA requirements from Mbed TLS capabilities */ -/****************************************************************/ - -#else /* MBEDTLS_PSA_CRYPTO_CONFIG */ - -/* - * Ensure PSA_WANT_* defines are setup properly if MBEDTLS_PSA_CRYPTO_CONFIG - * is not defined - */ - -#if defined(MBEDTLS_CCM_C) -#define MBEDTLS_PSA_BUILTIN_ALG_CCM 1 -#define PSA_WANT_ALG_CCM 1 -#endif /* MBEDTLS_CCM_C */ - -#if defined(MBEDTLS_CMAC_C) -#define MBEDTLS_PSA_BUILTIN_ALG_CMAC 1 -#define PSA_WANT_ALG_CMAC 1 -#endif /* MBEDTLS_CMAC_C */ - -#if defined(MBEDTLS_ECDH_C) -#define MBEDTLS_PSA_BUILTIN_ALG_ECDH 1 -#define PSA_WANT_ALG_ECDH 1 -#endif /* MBEDTLS_ECDH_C */ - -#if defined(MBEDTLS_ECDSA_C) -#define MBEDTLS_PSA_BUILTIN_ALG_ECDSA 1 -#define PSA_WANT_ALG_ECDSA 1 -#define PSA_WANT_ALG_ECDSA_ANY 1 - -// Only add in DETERMINISTIC support if ECDSA is also enabled -#if defined(MBEDTLS_ECDSA_DETERMINISTIC) -#define MBEDTLS_PSA_BUILTIN_ALG_DETERMINISTIC_ECDSA 1 -#define PSA_WANT_ALG_DETERMINISTIC_ECDSA 1 -#endif /* MBEDTLS_ECDSA_DETERMINISTIC */ - -#endif /* MBEDTLS_ECDSA_C */ - -#if defined(MBEDTLS_ECP_C) -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_ECC_KEY_PAIR 1 -#define PSA_WANT_KEY_TYPE_ECC_KEY_PAIR 1 -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_ECC_PUBLIC_KEY 1 -#define PSA_WANT_KEY_TYPE_ECC_PUBLIC_KEY 1 -#endif /* MBEDTLS_ECP_C */ - -#if defined(MBEDTLS_GCM_C) -#define MBEDTLS_PSA_BUILTIN_ALG_GCM 1 -#define PSA_WANT_ALG_GCM 1 -#endif /* MBEDTLS_GCM_C */ - -#if defined(MBEDTLS_HKDF_C) -#define MBEDTLS_PSA_BUILTIN_ALG_HMAC 1 -#define PSA_WANT_ALG_HMAC 1 -#define MBEDTLS_PSA_BUILTIN_ALG_HKDF 1 -#define PSA_WANT_ALG_HKDF 1 -#endif /* MBEDTLS_HKDF_C */ - -#if defined(MBEDTLS_MD_C) -#define MBEDTLS_PSA_BUILTIN_ALG_HMAC 1 -#define PSA_WANT_ALG_HMAC 1 -#define PSA_WANT_KEY_TYPE_HMAC -#define MBEDTLS_PSA_BUILTIN_ALG_TLS12_PRF 1 -#define PSA_WANT_ALG_TLS12_PRF 1 -#define MBEDTLS_PSA_BUILTIN_ALG_TLS12_PSK_TO_MS 1 -#define PSA_WANT_ALG_TLS12_PSK_TO_MS 1 -#endif /* MBEDTLS_MD_C */ - -#if defined(MBEDTLS_MD5_C) -#define MBEDTLS_PSA_BUILTIN_ALG_MD5 1 -#define PSA_WANT_ALG_MD5 1 -#endif - -#if defined(MBEDTLS_RIPEMD160_C) -#define MBEDTLS_PSA_BUILTIN_ALG_RIPEMD160 1 -#define PSA_WANT_ALG_RIPEMD160 1 -#endif - -#if defined(MBEDTLS_RSA_C) -#if defined(MBEDTLS_PKCS1_V15) -#define MBEDTLS_PSA_BUILTIN_ALG_RSA_PKCS1V15_CRYPT 1 -#define PSA_WANT_ALG_RSA_PKCS1V15_CRYPT 1 -#define MBEDTLS_PSA_BUILTIN_ALG_RSA_PKCS1V15_SIGN 1 -#define PSA_WANT_ALG_RSA_PKCS1V15_SIGN 1 -#define PSA_WANT_ALG_RSA_PKCS1V15_SIGN_RAW 1 -#endif /* MBEDTLSS_PKCS1_V15 */ -#if defined(MBEDTLS_PKCS1_V21) -#define MBEDTLS_PSA_BUILTIN_ALG_RSA_OAEP 1 -#define PSA_WANT_ALG_RSA_OAEP 1 -#define MBEDTLS_PSA_BUILTIN_ALG_RSA_PSS 1 -#define PSA_WANT_ALG_RSA_PSS 1 -#endif /* MBEDTLS_PKCS1_V21 */ -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_RSA_KEY_PAIR 1 -#define PSA_WANT_KEY_TYPE_RSA_KEY_PAIR 1 -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_RSA_PUBLIC_KEY 1 -#define PSA_WANT_KEY_TYPE_RSA_PUBLIC_KEY 1 -#endif /* MBEDTLS_RSA_C */ - -#if defined(MBEDTLS_SHA1_C) -#define MBEDTLS_PSA_BUILTIN_ALG_SHA_1 1 -#define PSA_WANT_ALG_SHA_1 1 -#endif - -#if defined(MBEDTLS_SHA224_C) -#define MBEDTLS_PSA_BUILTIN_ALG_SHA_224 1 -#define PSA_WANT_ALG_SHA_224 1 -#endif - -#if defined(MBEDTLS_SHA256_C) -#define MBEDTLS_PSA_BUILTIN_ALG_SHA_256 1 -#define PSA_WANT_ALG_SHA_256 1 -#endif - -#if defined(MBEDTLS_SHA384_C) -#define MBEDTLS_PSA_BUILTIN_ALG_SHA_384 1 -#define PSA_WANT_ALG_SHA_384 1 -#endif - -#if defined(MBEDTLS_SHA512_C) -#define MBEDTLS_PSA_BUILTIN_ALG_SHA_512 1 -#define PSA_WANT_ALG_SHA_512 1 -#endif - -#if defined(MBEDTLS_AES_C) -#define PSA_WANT_KEY_TYPE_AES 1 -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_AES 1 -#endif - -#if defined(MBEDTLS_CAMELLIA_C) -#define PSA_WANT_KEY_TYPE_CAMELLIA 1 -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_CAMELLIA 1 -#endif - -#if defined(MBEDTLS_DES_C) -#define PSA_WANT_KEY_TYPE_DES 1 -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_DES 1 -#endif - -#if defined(MBEDTLS_CHACHA20_C) -#define PSA_WANT_KEY_TYPE_CHACHA20 1 -#define PSA_WANT_ALG_STREAM_CIPHER 1 -#define MBEDTLS_PSA_BUILTIN_KEY_TYPE_CHACHA20 1 -#define MBEDTLS_PSA_BUILTIN_ALG_STREAM_CIPHER 1 -#if defined(MBEDTLS_CHACHAPOLY_C) -#define PSA_WANT_ALG_CHACHA20_POLY1305 1 -#define MBEDTLS_PSA_BUILTIN_ALG_CHACHA20_POLY1305 1 -#endif -#endif - -#if defined(MBEDTLS_CIPHER_MODE_CBC) -#define MBEDTLS_PSA_BUILTIN_ALG_CBC_NO_PADDING 1 -#define PSA_WANT_ALG_CBC_NO_PADDING 1 -#if defined(MBEDTLS_CIPHER_PADDING_PKCS7) -#define MBEDTLS_PSA_BUILTIN_ALG_CBC_PKCS7 1 -#define PSA_WANT_ALG_CBC_PKCS7 1 -#endif -#endif - -#if defined(MBEDTLS_AES_C) || defined(MBEDTLS_DES_C) || \ - defined(MBEDTLS_CAMELLIA_C) -#define MBEDTLS_PSA_BUILTIN_ALG_ECB_NO_PADDING 1 -#define PSA_WANT_ALG_ECB_NO_PADDING 1 -#endif - -#if defined(MBEDTLS_CIPHER_MODE_CFB) -#define MBEDTLS_PSA_BUILTIN_ALG_CFB 1 -#define PSA_WANT_ALG_CFB 1 -#endif - -#if defined(MBEDTLS_CIPHER_MODE_CTR) -#define MBEDTLS_PSA_BUILTIN_ALG_CTR 1 -#define PSA_WANT_ALG_CTR 1 -#endif - -#if defined(MBEDTLS_CIPHER_MODE_OFB) -#define MBEDTLS_PSA_BUILTIN_ALG_OFB 1 -#define PSA_WANT_ALG_OFB 1 -#endif - -#if defined(MBEDTLS_CIPHER_MODE_XTS) -#define MBEDTLS_PSA_BUILTIN_ALG_XTS 1 -#define PSA_WANT_ALG_XTS 1 -#endif - -#if defined(MBEDTLS_ECP_DP_BP256R1_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_BRAINPOOL_P_R1_256 1 -#define PSA_WANT_ECC_BRAINPOOL_P_R1_256 -#endif - -#if defined(MBEDTLS_ECP_DP_BP384R1_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_BRAINPOOL_P_R1_384 1 -#define PSA_WANT_ECC_BRAINPOOL_P_R1_384 -#endif - -#if defined(MBEDTLS_ECP_DP_BP512R1_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_BRAINPOOL_P_R1_512 1 -#define PSA_WANT_ECC_BRAINPOOL_P_R1_512 -#endif - -#if defined(MBEDTLS_ECP_DP_CURVE25519_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_MONTGOMERY_255 1 -#define PSA_WANT_ECC_MONTGOMERY_255 -#endif - -/* Curve448 is not yet supported via the PSA API (https://github.com/ARMmbed/mbedtls/issues/4249) */ -#if 0 && defined(MBEDTLS_ECP_DP_CURVE448_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_MONTGOMERY_448 1 -#define PSA_WANT_ECC_MONTGOMERY_448 -#endif - -#if defined(MBEDTLS_ECP_DP_SECP192R1_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_R1_192 1 -#define PSA_WANT_ECC_SECP_R1_192 -#endif - -#if defined(MBEDTLS_ECP_DP_SECP224R1_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_R1_224 1 -#define PSA_WANT_ECC_SECP_R1_224 -#endif - -#if defined(MBEDTLS_ECP_DP_SECP256R1_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_R1_256 1 -#define PSA_WANT_ECC_SECP_R1_256 -#endif - -#if defined(MBEDTLS_ECP_DP_SECP384R1_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_R1_384 1 -#define PSA_WANT_ECC_SECP_R1_384 -#endif - -#if defined(MBEDTLS_ECP_DP_SECP521R1_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_R1_521 1 -#define PSA_WANT_ECC_SECP_R1_521 -#endif - -#if defined(MBEDTLS_ECP_DP_SECP192K1_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_K1_192 1 -#define PSA_WANT_ECC_SECP_K1_192 -#endif - -/* SECP224K1 is buggy via the PSA API (https://github.com/ARMmbed/mbedtls/issues/3541) */ -#if 0 && defined(MBEDTLS_ECP_DP_SECP224K1_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_K1_224 1 -#define PSA_WANT_ECC_SECP_K1_224 -#endif - -#if defined(MBEDTLS_ECP_DP_SECP256K1_ENABLED) -#define MBEDTLS_PSA_BUILTIN_ECC_SECP_K1_256 1 -#define PSA_WANT_ECC_SECP_K1_256 -#endif - -#endif /* MBEDTLS_PSA_CRYPTO_CONFIG */ - -/* These features are always enabled. */ -#define PSA_WANT_KEY_TYPE_DERIVE 1 -#define PSA_WANT_KEY_TYPE_RAW_DATA 1 - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_CONFIG_PSA_H */ diff --git a/windows/mbedtls/ctr_drbg.h b/windows/mbedtls/ctr_drbg.h deleted file mode 100644 index c188b08a..00000000 --- a/windows/mbedtls/ctr_drbg.h +++ /dev/null @@ -1,575 +0,0 @@ -/** - * \file ctr_drbg.h - * - * \brief This file contains definitions and functions for the - * CTR_DRBG pseudorandom generator. - * - * CTR_DRBG is a standardized way of building a PRNG from a block-cipher - * in counter mode operation, as defined in NIST SP 800-90A: - * Recommendation for Random Number Generation Using Deterministic Random - * Bit Generators. - * - * The Mbed TLS implementation of CTR_DRBG uses AES-256 (default) or AES-128 - * (if \c MBEDTLS_CTR_DRBG_USE_128_BIT_KEY is enabled at compile time) - * as the underlying block cipher, with a derivation function. - * - * The security strength as defined in NIST SP 800-90A is - * 128 bits when AES-128 is used (\c MBEDTLS_CTR_DRBG_USE_128_BIT_KEY enabled) - * and 256 bits otherwise, provided that #MBEDTLS_CTR_DRBG_ENTROPY_LEN is - * kept at its default value (and not overridden in mbedtls_config.h) and that the - * DRBG instance is set up with default parameters. - * See the documentation of mbedtls_ctr_drbg_seed() for more - * information. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_CTR_DRBG_H -#define MBEDTLS_CTR_DRBG_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/aes.h" - -#if defined(MBEDTLS_THREADING_C) -#include "mbedtls/threading.h" -#endif - -#define MBEDTLS_ERR_CTR_DRBG_ENTROPY_SOURCE_FAILED -0x0034 /**< The entropy source failed. */ -#define MBEDTLS_ERR_CTR_DRBG_REQUEST_TOO_BIG -0x0036 /**< The requested random buffer length is too big. */ -#define MBEDTLS_ERR_CTR_DRBG_INPUT_TOO_BIG -0x0038 /**< The input (entropy + additional data) is too large. */ -#define MBEDTLS_ERR_CTR_DRBG_FILE_IO_ERROR -0x003A /**< Read or write error in file. */ - -#define MBEDTLS_CTR_DRBG_BLOCKSIZE 16 /**< The block size used by the cipher. */ - -#if defined(MBEDTLS_CTR_DRBG_USE_128_BIT_KEY) -#define MBEDTLS_CTR_DRBG_KEYSIZE 16 -/**< The key size in bytes used by the cipher. - * - * Compile-time choice: 16 bytes (128 bits) - * because #MBEDTLS_CTR_DRBG_USE_128_BIT_KEY is enabled. - */ -#else -#define MBEDTLS_CTR_DRBG_KEYSIZE 32 -/**< The key size in bytes used by the cipher. - * - * Compile-time choice: 32 bytes (256 bits) - * because \c MBEDTLS_CTR_DRBG_USE_128_BIT_KEY is disabled. - */ -#endif - -#define MBEDTLS_CTR_DRBG_KEYBITS ( MBEDTLS_CTR_DRBG_KEYSIZE * 8 ) /**< The key size for the DRBG operation, in bits. */ -#define MBEDTLS_CTR_DRBG_SEEDLEN ( MBEDTLS_CTR_DRBG_KEYSIZE + MBEDTLS_CTR_DRBG_BLOCKSIZE ) /**< The seed length, calculated as (counter + AES key). */ - -/** - * \name SECTION: Module settings - * - * The configuration options you can set for this module are in this section. - * Either change them in mbedtls_config.h or define them using the compiler command - * line. - * \{ - */ - -/** \def MBEDTLS_CTR_DRBG_ENTROPY_LEN - * - * \brief The amount of entropy used per seed by default, in bytes. - */ -#if !defined(MBEDTLS_CTR_DRBG_ENTROPY_LEN) -#if defined(MBEDTLS_SHA512_C) && !defined(MBEDTLS_ENTROPY_FORCE_SHA256) -/** This is 48 bytes because the entropy module uses SHA-512 - * (\c MBEDTLS_ENTROPY_FORCE_SHA256 is disabled). - */ -#define MBEDTLS_CTR_DRBG_ENTROPY_LEN 48 - -#else /* defined(MBEDTLS_SHA512_C) && !defined(MBEDTLS_ENTROPY_FORCE_SHA256) */ - -/** This is 32 bytes because the entropy module uses SHA-256 - * (the SHA512 module is disabled or - * \c MBEDTLS_ENTROPY_FORCE_SHA256 is enabled). - */ -#if !defined(MBEDTLS_CTR_DRBG_USE_128_BIT_KEY) -/** \warning To achieve a 256-bit security strength, you must pass a nonce - * to mbedtls_ctr_drbg_seed(). - */ -#endif /* !defined(MBEDTLS_CTR_DRBG_USE_128_BIT_KEY) */ -#define MBEDTLS_CTR_DRBG_ENTROPY_LEN 32 -#endif /* defined(MBEDTLS_SHA512_C) && !defined(MBEDTLS_ENTROPY_FORCE_SHA256) */ -#endif /* !defined(MBEDTLS_CTR_DRBG_ENTROPY_LEN) */ - -#if !defined(MBEDTLS_CTR_DRBG_RESEED_INTERVAL) -#define MBEDTLS_CTR_DRBG_RESEED_INTERVAL 10000 -/**< The interval before reseed is performed by default. */ -#endif - -#if !defined(MBEDTLS_CTR_DRBG_MAX_INPUT) -#define MBEDTLS_CTR_DRBG_MAX_INPUT 256 -/**< The maximum number of additional input Bytes. */ -#endif - -#if !defined(MBEDTLS_CTR_DRBG_MAX_REQUEST) -#define MBEDTLS_CTR_DRBG_MAX_REQUEST 1024 -/**< The maximum number of requested Bytes per call. */ -#endif - -#if !defined(MBEDTLS_CTR_DRBG_MAX_SEED_INPUT) -#define MBEDTLS_CTR_DRBG_MAX_SEED_INPUT 384 -/**< The maximum size of seed or reseed buffer in bytes. */ -#endif - -/* \} name SECTION: Module settings */ - -#define MBEDTLS_CTR_DRBG_PR_OFF 0 -/**< Prediction resistance is disabled. */ -#define MBEDTLS_CTR_DRBG_PR_ON 1 -/**< Prediction resistance is enabled. */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if MBEDTLS_CTR_DRBG_ENTROPY_LEN >= MBEDTLS_CTR_DRBG_KEYSIZE * 3 / 2 -/** The default length of the nonce read from the entropy source. - * - * This is \c 0 because a single read from the entropy source is sufficient - * to include a nonce. - * See the documentation of mbedtls_ctr_drbg_seed() for more information. - */ -#define MBEDTLS_CTR_DRBG_ENTROPY_NONCE_LEN 0 -#else -/** The default length of the nonce read from the entropy source. - * - * This is half of the default entropy length because a single read from - * the entropy source does not provide enough material to form a nonce. - * See the documentation of mbedtls_ctr_drbg_seed() for more information. - */ -#define MBEDTLS_CTR_DRBG_ENTROPY_NONCE_LEN ( MBEDTLS_CTR_DRBG_ENTROPY_LEN + 1 ) / 2 -#endif - -/** - * \brief The CTR_DRBG context structure. - */ -typedef struct mbedtls_ctr_drbg_context -{ - unsigned char MBEDTLS_PRIVATE(counter)[16]; /*!< The counter (V). */ - int MBEDTLS_PRIVATE(reseed_counter); /*!< The reseed counter. - * This is the number of requests that have - * been made since the last (re)seeding, - * minus one. - * Before the initial seeding, this field - * contains the amount of entropy in bytes - * to use as a nonce for the initial seeding, - * or -1 if no nonce length has been explicitly - * set (see mbedtls_ctr_drbg_set_nonce_len()). - */ - int MBEDTLS_PRIVATE(prediction_resistance); /*!< This determines whether prediction - resistance is enabled, that is - whether to systematically reseed before - each random generation. */ - size_t MBEDTLS_PRIVATE(entropy_len); /*!< The amount of entropy grabbed on each - seed or reseed operation, in bytes. */ - int MBEDTLS_PRIVATE(reseed_interval); /*!< The reseed interval. - * This is the maximum number of requests - * that can be made between reseedings. */ - - mbedtls_aes_context MBEDTLS_PRIVATE(aes_ctx); /*!< The AES context. */ - - /* - * Callbacks (Entropy) - */ - int (*MBEDTLS_PRIVATE(f_entropy))(void *, unsigned char *, size_t); - /*!< The entropy callback function. */ - - void *MBEDTLS_PRIVATE(p_entropy); /*!< The context for the entropy function. */ - -#if defined(MBEDTLS_THREADING_C) - /* Invariant: the mutex is initialized if and only if f_entropy != NULL. - * This means that the mutex is initialized during the initial seeding - * in mbedtls_ctr_drbg_seed() and freed in mbedtls_ctr_drbg_free(). - * - * Note that this invariant may change without notice. Do not rely on it - * and do not access the mutex directly in application code. - */ - mbedtls_threading_mutex_t MBEDTLS_PRIVATE(mutex); -#endif -} -mbedtls_ctr_drbg_context; - -/** - * \brief This function initializes the CTR_DRBG context, - * and prepares it for mbedtls_ctr_drbg_seed() - * or mbedtls_ctr_drbg_free(). - * - * \note The reseed interval is - * #MBEDTLS_CTR_DRBG_RESEED_INTERVAL by default. - * You can override it by calling - * mbedtls_ctr_drbg_set_reseed_interval(). - * - * \param ctx The CTR_DRBG context to initialize. - */ -void mbedtls_ctr_drbg_init( mbedtls_ctr_drbg_context *ctx ); - -/** - * \brief This function seeds and sets up the CTR_DRBG - * entropy source for future reseeds. - * - * A typical choice for the \p f_entropy and \p p_entropy parameters is - * to use the entropy module: - * - \p f_entropy is mbedtls_entropy_func(); - * - \p p_entropy is an instance of ::mbedtls_entropy_context initialized - * with mbedtls_entropy_init() (which registers the platform's default - * entropy sources). - * - * The entropy length is #MBEDTLS_CTR_DRBG_ENTROPY_LEN by default. - * You can override it by calling mbedtls_ctr_drbg_set_entropy_len(). - * - * The entropy nonce length is: - * - \c 0 if the entropy length is at least 3/2 times the entropy length, - * which guarantees that the security strength is the maximum permitted - * by the key size and entropy length according to NIST SP 800-90A §10.2.1; - * - Half the entropy length otherwise. - * You can override it by calling mbedtls_ctr_drbg_set_nonce_len(). - * With the default entropy length, the entropy nonce length is - * #MBEDTLS_CTR_DRBG_ENTROPY_NONCE_LEN. - * - * You can provide a nonce and personalization string in addition to the - * entropy source, to make this instantiation as unique as possible. - * See SP 800-90A §8.6.7 for more details about nonces. - * - * The _seed_material_ value passed to the derivation function in - * the CTR_DRBG Instantiate Process described in NIST SP 800-90A §10.2.1.3.2 - * is the concatenation of the following strings: - * - A string obtained by calling \p f_entropy function for the entropy - * length. - */ -#if MBEDTLS_CTR_DRBG_ENTROPY_NONCE_LEN == 0 -/** - * - If mbedtls_ctr_drbg_set_nonce_len() has been called, a string - * obtained by calling \p f_entropy function for the specified length. - */ -#else -/** - * - A string obtained by calling \p f_entropy function for the entropy nonce - * length. If the entropy nonce length is \c 0, this function does not - * make a second call to \p f_entropy. - */ -#endif -#if defined(MBEDTLS_THREADING_C) -/** - * \note When Mbed TLS is built with threading support, - * after this function returns successfully, - * it is safe to call mbedtls_ctr_drbg_random() - * from multiple threads. Other operations, including - * reseeding, are not thread-safe. - */ -#endif /* MBEDTLS_THREADING_C */ -/** - * - The \p custom string. - * - * \note To achieve the nominal security strength permitted - * by CTR_DRBG, the entropy length must be: - * - at least 16 bytes for a 128-bit strength - * (maximum achievable strength when using AES-128); - * - at least 32 bytes for a 256-bit strength - * (maximum achievable strength when using AES-256). - * - * In addition, if you do not pass a nonce in \p custom, - * the sum of the entropy length - * and the entropy nonce length must be: - * - at least 24 bytes for a 128-bit strength - * (maximum achievable strength when using AES-128); - * - at least 48 bytes for a 256-bit strength - * (maximum achievable strength when using AES-256). - * - * \param ctx The CTR_DRBG context to seed. - * It must have been initialized with - * mbedtls_ctr_drbg_init(). - * After a successful call to mbedtls_ctr_drbg_seed(), - * you may not call mbedtls_ctr_drbg_seed() again on - * the same context unless you call - * mbedtls_ctr_drbg_free() and mbedtls_ctr_drbg_init() - * again first. - * After a failed call to mbedtls_ctr_drbg_seed(), - * you must call mbedtls_ctr_drbg_free(). - * \param f_entropy The entropy callback, taking as arguments the - * \p p_entropy context, the buffer to fill, and the - * length of the buffer. - * \p f_entropy is always called with a buffer size - * less than or equal to the entropy length. - * \param p_entropy The entropy context to pass to \p f_entropy. - * \param custom The personalization string. - * This can be \c NULL, in which case the personalization - * string is empty regardless of the value of \p len. - * \param len The length of the personalization string. - * This must be at most - * #MBEDTLS_CTR_DRBG_MAX_SEED_INPUT - * - #MBEDTLS_CTR_DRBG_ENTROPY_LEN. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CTR_DRBG_ENTROPY_SOURCE_FAILED on failure. - */ -int mbedtls_ctr_drbg_seed( mbedtls_ctr_drbg_context *ctx, - int (*f_entropy)(void *, unsigned char *, size_t), - void *p_entropy, - const unsigned char *custom, - size_t len ); - -/** - * \brief This function resets CTR_DRBG context to the state immediately - * after initial call of mbedtls_ctr_drbg_init(). - * - * \param ctx The CTR_DRBG context to clear. - */ -void mbedtls_ctr_drbg_free( mbedtls_ctr_drbg_context *ctx ); - -/** - * \brief This function turns prediction resistance on or off. - * The default value is off. - * - * \note If enabled, entropy is gathered at the beginning of - * every call to mbedtls_ctr_drbg_random_with_add() - * or mbedtls_ctr_drbg_random(). - * Only use this if your entropy source has sufficient - * throughput. - * - * \param ctx The CTR_DRBG context. - * \param resistance #MBEDTLS_CTR_DRBG_PR_ON or #MBEDTLS_CTR_DRBG_PR_OFF. - */ -void mbedtls_ctr_drbg_set_prediction_resistance( mbedtls_ctr_drbg_context *ctx, - int resistance ); - -/** - * \brief This function sets the amount of entropy grabbed on each - * seed or reseed. - * - * The default value is #MBEDTLS_CTR_DRBG_ENTROPY_LEN. - * - * \note The security strength of CTR_DRBG is bounded by the - * entropy length. Thus: - * - When using AES-256 - * (\c MBEDTLS_CTR_DRBG_USE_128_BIT_KEY is disabled, - * which is the default), - * \p len must be at least 32 (in bytes) - * to achieve a 256-bit strength. - * - When using AES-128 - * (\c MBEDTLS_CTR_DRBG_USE_128_BIT_KEY is enabled) - * \p len must be at least 16 (in bytes) - * to achieve a 128-bit strength. - * - * \param ctx The CTR_DRBG context. - * \param len The amount of entropy to grab, in bytes. - * This must be at most #MBEDTLS_CTR_DRBG_MAX_SEED_INPUT - * and at most the maximum length accepted by the - * entropy function that is set in the context. - */ -void mbedtls_ctr_drbg_set_entropy_len( mbedtls_ctr_drbg_context *ctx, - size_t len ); - -/** - * \brief This function sets the amount of entropy grabbed - * as a nonce for the initial seeding. - * - * Call this function before calling mbedtls_ctr_drbg_seed() to read - * a nonce from the entropy source during the initial seeding. - * - * \param ctx The CTR_DRBG context. - * \param len The amount of entropy to grab for the nonce, in bytes. - * This must be at most #MBEDTLS_CTR_DRBG_MAX_SEED_INPUT - * and at most the maximum length accepted by the - * entropy function that is set in the context. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CTR_DRBG_INPUT_TOO_BIG if \p len is - * more than #MBEDTLS_CTR_DRBG_MAX_SEED_INPUT. - * \return #MBEDTLS_ERR_CTR_DRBG_ENTROPY_SOURCE_FAILED - * if the initial seeding has already taken place. - */ -int mbedtls_ctr_drbg_set_nonce_len( mbedtls_ctr_drbg_context *ctx, - size_t len ); - -/** - * \brief This function sets the reseed interval. - * - * The reseed interval is the number of calls to mbedtls_ctr_drbg_random() - * or mbedtls_ctr_drbg_random_with_add() after which the entropy function - * is called again. - * - * The default value is #MBEDTLS_CTR_DRBG_RESEED_INTERVAL. - * - * \param ctx The CTR_DRBG context. - * \param interval The reseed interval. - */ -void mbedtls_ctr_drbg_set_reseed_interval( mbedtls_ctr_drbg_context *ctx, - int interval ); - -/** - * \brief This function reseeds the CTR_DRBG context, that is - * extracts data from the entropy source. - * - * \note This function is not thread-safe. It is not safe - * to call this function if another thread might be - * concurrently obtaining random numbers from the same - * context or updating or reseeding the same context. - * - * \param ctx The CTR_DRBG context. - * \param additional Additional data to add to the state. Can be \c NULL. - * \param len The length of the additional data. - * This must be less than - * #MBEDTLS_CTR_DRBG_MAX_SEED_INPUT - \c entropy_len - * where \c entropy_len is the entropy length - * configured for the context. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CTR_DRBG_ENTROPY_SOURCE_FAILED on failure. - */ -int mbedtls_ctr_drbg_reseed( mbedtls_ctr_drbg_context *ctx, - const unsigned char *additional, size_t len ); - -/** - * \brief This function updates the state of the CTR_DRBG context. - * - * \note This function is not thread-safe. It is not safe - * to call this function if another thread might be - * concurrently obtaining random numbers from the same - * context or updating or reseeding the same context. - * - * \param ctx The CTR_DRBG context. - * \param additional The data to update the state with. This must not be - * \c NULL unless \p add_len is \c 0. - * \param add_len Length of \p additional in bytes. This must be at - * most #MBEDTLS_CTR_DRBG_MAX_SEED_INPUT. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CTR_DRBG_INPUT_TOO_BIG if - * \p add_len is more than - * #MBEDTLS_CTR_DRBG_MAX_SEED_INPUT. - * \return An error from the underlying AES cipher on failure. - */ -int mbedtls_ctr_drbg_update( mbedtls_ctr_drbg_context *ctx, - const unsigned char *additional, - size_t add_len ); - -/** - * \brief This function updates a CTR_DRBG instance with additional - * data and uses it to generate random data. - * - * This function automatically reseeds if the reseed counter is exceeded - * or prediction resistance is enabled. - * - * \note This function is not thread-safe. It is not safe - * to call this function if another thread might be - * concurrently obtaining random numbers from the same - * context or updating or reseeding the same context. - * - * \param p_rng The CTR_DRBG context. This must be a pointer to a - * #mbedtls_ctr_drbg_context structure. - * \param output The buffer to fill. - * \param output_len The length of the buffer in bytes. - * \param additional Additional data to update. Can be \c NULL, in which - * case the additional data is empty regardless of - * the value of \p add_len. - * \param add_len The length of the additional data - * if \p additional is not \c NULL. - * This must be less than #MBEDTLS_CTR_DRBG_MAX_INPUT - * and less than - * #MBEDTLS_CTR_DRBG_MAX_SEED_INPUT - \c entropy_len - * where \c entropy_len is the entropy length - * configured for the context. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CTR_DRBG_ENTROPY_SOURCE_FAILED or - * #MBEDTLS_ERR_CTR_DRBG_REQUEST_TOO_BIG on failure. - */ -int mbedtls_ctr_drbg_random_with_add( void *p_rng, - unsigned char *output, size_t output_len, - const unsigned char *additional, size_t add_len ); - -/** - * \brief This function uses CTR_DRBG to generate random data. - * - * This function automatically reseeds if the reseed counter is exceeded - * or prediction resistance is enabled. - */ -#if defined(MBEDTLS_THREADING_C) -/** - * \note When Mbed TLS is built with threading support, - * it is safe to call mbedtls_ctr_drbg_random() - * from multiple threads. Other operations, including - * reseeding, are not thread-safe. - */ -#endif /* MBEDTLS_THREADING_C */ -/** - * \param p_rng The CTR_DRBG context. This must be a pointer to a - * #mbedtls_ctr_drbg_context structure. - * \param output The buffer to fill. - * \param output_len The length of the buffer in bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CTR_DRBG_ENTROPY_SOURCE_FAILED or - * #MBEDTLS_ERR_CTR_DRBG_REQUEST_TOO_BIG on failure. - */ -int mbedtls_ctr_drbg_random( void *p_rng, - unsigned char *output, size_t output_len ); - -#if defined(MBEDTLS_FS_IO) -/** - * \brief This function writes a seed file. - * - * \param ctx The CTR_DRBG context. - * \param path The name of the file. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CTR_DRBG_FILE_IO_ERROR on file error. - * \return #MBEDTLS_ERR_CTR_DRBG_ENTROPY_SOURCE_FAILED on reseed - * failure. - */ -int mbedtls_ctr_drbg_write_seed_file( mbedtls_ctr_drbg_context *ctx, const char *path ); - -/** - * \brief This function reads and updates a seed file. The seed - * is added to this instance. - * - * \param ctx The CTR_DRBG context. - * \param path The name of the file. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_CTR_DRBG_FILE_IO_ERROR on file error. - * \return #MBEDTLS_ERR_CTR_DRBG_ENTROPY_SOURCE_FAILED on - * reseed failure. - * \return #MBEDTLS_ERR_CTR_DRBG_INPUT_TOO_BIG if the existing - * seed file is too large. - */ -int mbedtls_ctr_drbg_update_seed_file( mbedtls_ctr_drbg_context *ctx, const char *path ); -#endif /* MBEDTLS_FS_IO */ - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief The CTR_DRBG checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_ctr_drbg_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* ctr_drbg.h */ diff --git a/windows/mbedtls/debug.h b/windows/mbedtls/debug.h deleted file mode 100644 index 0aed5961..00000000 --- a/windows/mbedtls/debug.h +++ /dev/null @@ -1,311 +0,0 @@ -/** - * \file debug.h - * - * \brief Functions for controlling and providing debug output from the library. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_DEBUG_H -#define MBEDTLS_DEBUG_H - -#include "mbedtls/build_info.h" - -#include "mbedtls/ssl.h" - -#if defined(MBEDTLS_ECP_C) -#include "mbedtls/ecp.h" -#endif - -#if defined(MBEDTLS_DEBUG_C) - -#define MBEDTLS_DEBUG_STRIP_PARENS( ... ) __VA_ARGS__ - -#define MBEDTLS_SSL_DEBUG_MSG( level, args ) \ - mbedtls_debug_print_msg( ssl, level, __FILE__, __LINE__, \ - MBEDTLS_DEBUG_STRIP_PARENS args ) - -#define MBEDTLS_SSL_DEBUG_RET( level, text, ret ) \ - mbedtls_debug_print_ret( ssl, level, __FILE__, __LINE__, text, ret ) - -#define MBEDTLS_SSL_DEBUG_BUF( level, text, buf, len ) \ - mbedtls_debug_print_buf( ssl, level, __FILE__, __LINE__, text, buf, len ) - -#if defined(MBEDTLS_BIGNUM_C) -#define MBEDTLS_SSL_DEBUG_MPI( level, text, X ) \ - mbedtls_debug_print_mpi( ssl, level, __FILE__, __LINE__, text, X ) -#endif - -#if defined(MBEDTLS_ECP_C) -#define MBEDTLS_SSL_DEBUG_ECP( level, text, X ) \ - mbedtls_debug_print_ecp( ssl, level, __FILE__, __LINE__, text, X ) -#endif - -#if defined(MBEDTLS_X509_CRT_PARSE_C) -#if !defined(MBEDTLS_X509_REMOVE_INFO) -#define MBEDTLS_SSL_DEBUG_CRT( level, text, crt ) \ - mbedtls_debug_print_crt( ssl, level, __FILE__, __LINE__, text, crt ) -#else -#define MBEDTLS_SSL_DEBUG_CRT( level, text, crt ) do { } while( 0 ) -#endif /* MBEDTLS_X509_REMOVE_INFO */ -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - -#if defined(MBEDTLS_ECDH_C) -#define MBEDTLS_SSL_DEBUG_ECDH( level, ecdh, attr ) \ - mbedtls_debug_printf_ecdh( ssl, level, __FILE__, __LINE__, ecdh, attr ) -#endif - -#else /* MBEDTLS_DEBUG_C */ - -#define MBEDTLS_SSL_DEBUG_MSG( level, args ) do { } while( 0 ) -#define MBEDTLS_SSL_DEBUG_RET( level, text, ret ) do { } while( 0 ) -#define MBEDTLS_SSL_DEBUG_BUF( level, text, buf, len ) do { } while( 0 ) -#define MBEDTLS_SSL_DEBUG_MPI( level, text, X ) do { } while( 0 ) -#define MBEDTLS_SSL_DEBUG_ECP( level, text, X ) do { } while( 0 ) -#define MBEDTLS_SSL_DEBUG_CRT( level, text, crt ) do { } while( 0 ) -#define MBEDTLS_SSL_DEBUG_ECDH( level, ecdh, attr ) do { } while( 0 ) - -#endif /* MBEDTLS_DEBUG_C */ - -/** - * \def MBEDTLS_PRINTF_ATTRIBUTE - * - * Mark a function as having printf attributes, and thus enable checking - * via -wFormat and other flags. This does nothing on builds with compilers - * that do not support the format attribute - * - * Module: library/debug.c - * Caller: - * - * This module provides debugging functions. - */ -#if defined(__has_attribute) -#if __has_attribute(format) -#if defined(__MINGW32__) && __USE_MINGW_ANSI_STDIO == 1 -#define MBEDTLS_PRINTF_ATTRIBUTE(string_index, first_to_check) \ - __attribute__((__format__ (gnu_printf, string_index, first_to_check))) -#else /* defined(__MINGW32__) && __USE_MINGW_ANSI_STDIO == 1 */ -#define MBEDTLS_PRINTF_ATTRIBUTE(string_index, first_to_check) \ - __attribute__((format(printf, string_index, first_to_check))) -#endif -#else /* __has_attribute(format) */ -#define MBEDTLS_PRINTF_ATTRIBUTE(string_index, first_to_check) -#endif /* __has_attribute(format) */ -#else /* defined(__has_attribute) */ -#define MBEDTLS_PRINTF_ATTRIBUTE(string_index, first_to_check) -#endif - -/** - * \def MBEDTLS_PRINTF_SIZET - * - * MBEDTLS_PRINTF_xxx: Due to issues with older window compilers - * and MinGW we need to define the printf specifier for size_t - * and long long per platform. - * - * Module: library/debug.c - * Caller: - * - * This module provides debugging functions. - */ -#if (defined(__MINGW32__) && __USE_MINGW_ANSI_STDIO == 0) || (defined(_MSC_VER) && _MSC_VER < 1800) - #include - #define MBEDTLS_PRINTF_SIZET PRIuPTR - #define MBEDTLS_PRINTF_LONGLONG "I64d" -#else /* (defined(__MINGW32__) && __USE_MINGW_ANSI_STDIO == 0) || (defined(_MSC_VER) && _MSC_VER < 1800) */ - #define MBEDTLS_PRINTF_SIZET "zu" - #define MBEDTLS_PRINTF_LONGLONG "lld" -#endif /* (defined(__MINGW32__) && __USE_MINGW_ANSI_STDIO == 0) || (defined(_MSC_VER) && _MSC_VER < 1800) */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief Set the threshold error level to handle globally all debug output. - * Debug messages that have a level over the threshold value are - * discarded. - * (Default value: 0 = No debug ) - * - * \param threshold theshold level of messages to filter on. Messages at a - * higher level will be discarded. - * - Debug levels - * - 0 No debug - * - 1 Error - * - 2 State change - * - 3 Informational - * - 4 Verbose - */ -void mbedtls_debug_set_threshold( int threshold ); - -/** - * \brief Print a message to the debug output. This function is always used - * through the MBEDTLS_SSL_DEBUG_MSG() macro, which supplies the ssl - * context, file and line number parameters. - * - * \param ssl SSL context - * \param level error level of the debug message - * \param file file the message has occurred in - * \param line line number the message has occurred at - * \param format format specifier, in printf format - * \param ... variables used by the format specifier - * - * \attention This function is intended for INTERNAL usage within the - * library only. - */ -void mbedtls_debug_print_msg( const mbedtls_ssl_context *ssl, int level, - const char *file, int line, - const char *format, ... ) MBEDTLS_PRINTF_ATTRIBUTE(5, 6); - -/** - * \brief Print the return value of a function to the debug output. This - * function is always used through the MBEDTLS_SSL_DEBUG_RET() macro, - * which supplies the ssl context, file and line number parameters. - * - * \param ssl SSL context - * \param level error level of the debug message - * \param file file the error has occurred in - * \param line line number the error has occurred in - * \param text the name of the function that returned the error - * \param ret the return code value - * - * \attention This function is intended for INTERNAL usage within the - * library only. - */ -void mbedtls_debug_print_ret( const mbedtls_ssl_context *ssl, int level, - const char *file, int line, - const char *text, int ret ); - -/** - * \brief Output a buffer of size len bytes to the debug output. This function - * is always used through the MBEDTLS_SSL_DEBUG_BUF() macro, - * which supplies the ssl context, file and line number parameters. - * - * \param ssl SSL context - * \param level error level of the debug message - * \param file file the error has occurred in - * \param line line number the error has occurred in - * \param text a name or label for the buffer being dumped. Normally the - * variable or buffer name - * \param buf the buffer to be outputted - * \param len length of the buffer - * - * \attention This function is intended for INTERNAL usage within the - * library only. - */ -void mbedtls_debug_print_buf( const mbedtls_ssl_context *ssl, int level, - const char *file, int line, const char *text, - const unsigned char *buf, size_t len ); - -#if defined(MBEDTLS_BIGNUM_C) -/** - * \brief Print a MPI variable to the debug output. This function is always - * used through the MBEDTLS_SSL_DEBUG_MPI() macro, which supplies the - * ssl context, file and line number parameters. - * - * \param ssl SSL context - * \param level error level of the debug message - * \param file file the error has occurred in - * \param line line number the error has occurred in - * \param text a name or label for the MPI being output. Normally the - * variable name - * \param X the MPI variable - * - * \attention This function is intended for INTERNAL usage within the - * library only. - */ -void mbedtls_debug_print_mpi( const mbedtls_ssl_context *ssl, int level, - const char *file, int line, - const char *text, const mbedtls_mpi *X ); -#endif - -#if defined(MBEDTLS_ECP_C) -/** - * \brief Print an ECP point to the debug output. This function is always - * used through the MBEDTLS_SSL_DEBUG_ECP() macro, which supplies the - * ssl context, file and line number parameters. - * - * \param ssl SSL context - * \param level error level of the debug message - * \param file file the error has occurred in - * \param line line number the error has occurred in - * \param text a name or label for the ECP point being output. Normally the - * variable name - * \param X the ECP point - * - * \attention This function is intended for INTERNAL usage within the - * library only. - */ -void mbedtls_debug_print_ecp( const mbedtls_ssl_context *ssl, int level, - const char *file, int line, - const char *text, const mbedtls_ecp_point *X ); -#endif - -#if defined(MBEDTLS_X509_CRT_PARSE_C) && !defined(MBEDTLS_X509_REMOVE_INFO) -/** - * \brief Print a X.509 certificate structure to the debug output. This - * function is always used through the MBEDTLS_SSL_DEBUG_CRT() macro, - * which supplies the ssl context, file and line number parameters. - * - * \param ssl SSL context - * \param level error level of the debug message - * \param file file the error has occurred in - * \param line line number the error has occurred in - * \param text a name or label for the certificate being output - * \param crt X.509 certificate structure - * - * \attention This function is intended for INTERNAL usage within the - * library only. - */ -void mbedtls_debug_print_crt( const mbedtls_ssl_context *ssl, int level, - const char *file, int line, - const char *text, const mbedtls_x509_crt *crt ); -#endif - -#if defined(MBEDTLS_ECDH_C) -typedef enum -{ - MBEDTLS_DEBUG_ECDH_Q, - MBEDTLS_DEBUG_ECDH_QP, - MBEDTLS_DEBUG_ECDH_Z, -} mbedtls_debug_ecdh_attr; - -/** - * \brief Print a field of the ECDH structure in the SSL context to the debug - * output. This function is always used through the - * MBEDTLS_SSL_DEBUG_ECDH() macro, which supplies the ssl context, file - * and line number parameters. - * - * \param ssl SSL context - * \param level error level of the debug message - * \param file file the error has occurred in - * \param line line number the error has occurred in - * \param ecdh the ECDH context - * \param attr the identifier of the attribute being output - * - * \attention This function is intended for INTERNAL usage within the - * library only. - */ -void mbedtls_debug_printf_ecdh( const mbedtls_ssl_context *ssl, int level, - const char *file, int line, - const mbedtls_ecdh_context *ecdh, - mbedtls_debug_ecdh_attr attr ); -#endif - -#ifdef __cplusplus -} -#endif - -#endif /* debug.h */ diff --git a/windows/mbedtls/des.h b/windows/mbedtls/des.h deleted file mode 100644 index 9e2f6839..00000000 --- a/windows/mbedtls/des.h +++ /dev/null @@ -1,348 +0,0 @@ -/** - * \file des.h - * - * \brief DES block cipher - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - * - */ -#ifndef MBEDTLS_DES_H -#define MBEDTLS_DES_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -#define MBEDTLS_DES_ENCRYPT 1 -#define MBEDTLS_DES_DECRYPT 0 - -#define MBEDTLS_ERR_DES_INVALID_INPUT_LENGTH -0x0032 /**< The data input has an invalid length. */ - -#define MBEDTLS_DES_KEY_SIZE 8 - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_DES_ALT) -// Regular implementation -// - -/** - * \brief DES context structure - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -typedef struct mbedtls_des_context -{ - uint32_t MBEDTLS_PRIVATE(sk)[32]; /*!< DES subkeys */ -} -mbedtls_des_context; - -/** - * \brief Triple-DES context structure - */ -typedef struct mbedtls_des3_context -{ - uint32_t MBEDTLS_PRIVATE(sk)[96]; /*!< 3DES subkeys */ -} -mbedtls_des3_context; - -#else /* MBEDTLS_DES_ALT */ -#include "des_alt.h" -#endif /* MBEDTLS_DES_ALT */ - -/** - * \brief Initialize DES context - * - * \param ctx DES context to be initialized - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -void mbedtls_des_init( mbedtls_des_context *ctx ); - -/** - * \brief Clear DES context - * - * \param ctx DES context to be cleared - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -void mbedtls_des_free( mbedtls_des_context *ctx ); - -/** - * \brief Initialize Triple-DES context - * - * \param ctx DES3 context to be initialized - */ -void mbedtls_des3_init( mbedtls_des3_context *ctx ); - -/** - * \brief Clear Triple-DES context - * - * \param ctx DES3 context to be cleared - */ -void mbedtls_des3_free( mbedtls_des3_context *ctx ); - -/** - * \brief Set key parity on the given key to odd. - * - * DES keys are 56 bits long, but each byte is padded with - * a parity bit to allow verification. - * - * \param key 8-byte secret key - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -void mbedtls_des_key_set_parity( unsigned char key[MBEDTLS_DES_KEY_SIZE] ); - -/** - * \brief Check that key parity on the given key is odd. - * - * DES keys are 56 bits long, but each byte is padded with - * a parity bit to allow verification. - * - * \param key 8-byte secret key - * - * \return 0 is parity was ok, 1 if parity was not correct. - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -int mbedtls_des_key_check_key_parity( const unsigned char key[MBEDTLS_DES_KEY_SIZE] ); - -/** - * \brief Check that key is not a weak or semi-weak DES key - * - * \param key 8-byte secret key - * - * \return 0 if no weak key was found, 1 if a weak key was identified. - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -int mbedtls_des_key_check_weak( const unsigned char key[MBEDTLS_DES_KEY_SIZE] ); - -/** - * \brief DES key schedule (56-bit, encryption) - * - * \param ctx DES context to be initialized - * \param key 8-byte secret key - * - * \return 0 - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -int mbedtls_des_setkey_enc( mbedtls_des_context *ctx, const unsigned char key[MBEDTLS_DES_KEY_SIZE] ); - -/** - * \brief DES key schedule (56-bit, decryption) - * - * \param ctx DES context to be initialized - * \param key 8-byte secret key - * - * \return 0 - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -int mbedtls_des_setkey_dec( mbedtls_des_context *ctx, const unsigned char key[MBEDTLS_DES_KEY_SIZE] ); - -/** - * \brief Triple-DES key schedule (112-bit, encryption) - * - * \param ctx 3DES context to be initialized - * \param key 16-byte secret key - * - * \return 0 - */ -int mbedtls_des3_set2key_enc( mbedtls_des3_context *ctx, - const unsigned char key[MBEDTLS_DES_KEY_SIZE * 2] ); - -/** - * \brief Triple-DES key schedule (112-bit, decryption) - * - * \param ctx 3DES context to be initialized - * \param key 16-byte secret key - * - * \return 0 - */ -int mbedtls_des3_set2key_dec( mbedtls_des3_context *ctx, - const unsigned char key[MBEDTLS_DES_KEY_SIZE * 2] ); - -/** - * \brief Triple-DES key schedule (168-bit, encryption) - * - * \param ctx 3DES context to be initialized - * \param key 24-byte secret key - * - * \return 0 - */ -int mbedtls_des3_set3key_enc( mbedtls_des3_context *ctx, - const unsigned char key[MBEDTLS_DES_KEY_SIZE * 3] ); - -/** - * \brief Triple-DES key schedule (168-bit, decryption) - * - * \param ctx 3DES context to be initialized - * \param key 24-byte secret key - * - * \return 0 - */ -int mbedtls_des3_set3key_dec( mbedtls_des3_context *ctx, - const unsigned char key[MBEDTLS_DES_KEY_SIZE * 3] ); - -/** - * \brief DES-ECB block encryption/decryption - * - * \param ctx DES context - * \param input 64-bit input block - * \param output 64-bit output block - * - * \return 0 if successful - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -int mbedtls_des_crypt_ecb( mbedtls_des_context *ctx, - const unsigned char input[8], - unsigned char output[8] ); - -#if defined(MBEDTLS_CIPHER_MODE_CBC) -/** - * \brief DES-CBC buffer encryption/decryption - * - * \note Upon exit, the content of the IV is updated so that you can - * call the function same function again on the following - * block(s) of data and get the same result as if it was - * encrypted in one call. This allows a "streaming" usage. - * If on the other hand you need to retain the contents of the - * IV, you should either save it manually or use the cipher - * module instead. - * - * \param ctx DES context - * \param mode MBEDTLS_DES_ENCRYPT or MBEDTLS_DES_DECRYPT - * \param length length of the input data - * \param iv initialization vector (updated after use) - * \param input buffer holding the input data - * \param output buffer holding the output data - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -int mbedtls_des_crypt_cbc( mbedtls_des_context *ctx, - int mode, - size_t length, - unsigned char iv[8], - const unsigned char *input, - unsigned char *output ); -#endif /* MBEDTLS_CIPHER_MODE_CBC */ - -/** - * \brief 3DES-ECB block encryption/decryption - * - * \param ctx 3DES context - * \param input 64-bit input block - * \param output 64-bit output block - * - * \return 0 if successful - */ -int mbedtls_des3_crypt_ecb( mbedtls_des3_context *ctx, - const unsigned char input[8], - unsigned char output[8] ); - -#if defined(MBEDTLS_CIPHER_MODE_CBC) -/** - * \brief 3DES-CBC buffer encryption/decryption - * - * \note Upon exit, the content of the IV is updated so that you can - * call the function same function again on the following - * block(s) of data and get the same result as if it was - * encrypted in one call. This allows a "streaming" usage. - * If on the other hand you need to retain the contents of the - * IV, you should either save it manually or use the cipher - * module instead. - * - * \param ctx 3DES context - * \param mode MBEDTLS_DES_ENCRYPT or MBEDTLS_DES_DECRYPT - * \param length length of the input data - * \param iv initialization vector (updated after use) - * \param input buffer holding the input data - * \param output buffer holding the output data - * - * \return 0 if successful, or MBEDTLS_ERR_DES_INVALID_INPUT_LENGTH - */ -int mbedtls_des3_crypt_cbc( mbedtls_des3_context *ctx, - int mode, - size_t length, - unsigned char iv[8], - const unsigned char *input, - unsigned char *output ); -#endif /* MBEDTLS_CIPHER_MODE_CBC */ - -/** - * \brief Internal function for key expansion. - * (Only exposed to allow overriding it, - * see MBEDTLS_DES_SETKEY_ALT) - * - * \param SK Round keys - * \param key Base key - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers - * instead. - */ -void mbedtls_des_setkey( uint32_t SK[32], - const unsigned char key[MBEDTLS_DES_KEY_SIZE] ); - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief Checkup routine - * - * \return 0 if successful, or 1 if the test failed - */ -int mbedtls_des_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* des.h */ diff --git a/windows/mbedtls/dhm.h b/windows/mbedtls/dhm.h deleted file mode 100644 index 75baed95..00000000 --- a/windows/mbedtls/dhm.h +++ /dev/null @@ -1,976 +0,0 @@ -/** - * \file dhm.h - * - * \brief This file contains Diffie-Hellman-Merkle (DHM) key exchange - * definitions and functions. - * - * Diffie-Hellman-Merkle (DHM) key exchange is defined in - * RFC-2631: Diffie-Hellman Key Agreement Method and - * Public-Key Cryptography Standards (PKCS) #3: Diffie - * Hellman Key Agreement Standard. - * - * RFC-3526: More Modular Exponential (MODP) Diffie-Hellman groups for - * Internet Key Exchange (IKE) defines a number of standardized - * Diffie-Hellman groups for IKE. - * - * RFC-5114: Additional Diffie-Hellman Groups for Use with IETF - * Standards defines a number of standardized Diffie-Hellman - * groups that can be used. - * - * \warning The security of the DHM key exchange relies on the proper choice - * of prime modulus - optimally, it should be a safe prime. The usage - * of non-safe primes both decreases the difficulty of the underlying - * discrete logarithm problem and can lead to small subgroup attacks - * leaking private exponent bits when invalid public keys are used - * and not detected. This is especially relevant if the same DHM - * parameters are reused for multiple key exchanges as in static DHM, - * while the criticality of small-subgroup attacks is lower for - * ephemeral DHM. - * - * \warning For performance reasons, the code does neither perform primality - * nor safe primality tests, nor the expensive checks for invalid - * subgroups. Moreover, even if these were performed, non-standardized - * primes cannot be trusted because of the possibility of backdoors - * that can't be effectively checked for. - * - * \warning Diffie-Hellman-Merkle is therefore a security risk when not using - * standardized primes generated using a trustworthy ("nothing up - * my sleeve") method, such as the RFC 3526 / 7919 primes. In the TLS - * protocol, DH parameters need to be negotiated, so using the default - * primes systematically is not always an option. If possible, use - * Elliptic Curve Diffie-Hellman (ECDH), which has better performance, - * and for which the TLS protocol mandates the use of standard - * parameters. - * - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_DHM_H -#define MBEDTLS_DHM_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" -#include "mbedtls/bignum.h" - -/* - * DHM Error codes - */ -#define MBEDTLS_ERR_DHM_BAD_INPUT_DATA -0x3080 /**< Bad input parameters. */ -#define MBEDTLS_ERR_DHM_READ_PARAMS_FAILED -0x3100 /**< Reading of the DHM parameters failed. */ -#define MBEDTLS_ERR_DHM_MAKE_PARAMS_FAILED -0x3180 /**< Making of the DHM parameters failed. */ -#define MBEDTLS_ERR_DHM_READ_PUBLIC_FAILED -0x3200 /**< Reading of the public values failed. */ -#define MBEDTLS_ERR_DHM_MAKE_PUBLIC_FAILED -0x3280 /**< Making of the public value failed. */ -#define MBEDTLS_ERR_DHM_CALC_SECRET_FAILED -0x3300 /**< Calculation of the DHM secret failed. */ -#define MBEDTLS_ERR_DHM_INVALID_FORMAT -0x3380 /**< The ASN.1 data is not formatted correctly. */ -#define MBEDTLS_ERR_DHM_ALLOC_FAILED -0x3400 /**< Allocation of memory failed. */ -#define MBEDTLS_ERR_DHM_FILE_IO_ERROR -0x3480 /**< Read or write of file failed. */ -#define MBEDTLS_ERR_DHM_SET_GROUP_FAILED -0x3580 /**< Setting the modulus and generator failed. */ - -/** Which parameter to access in mbedtls_dhm_get_value(). */ -typedef enum -{ - MBEDTLS_DHM_PARAM_P, /*!< The prime modulus. */ - MBEDTLS_DHM_PARAM_G, /*!< The generator. */ - MBEDTLS_DHM_PARAM_X, /*!< Our secret value. */ - MBEDTLS_DHM_PARAM_GX, /*!< Our public key = \c G^X mod \c P. */ - MBEDTLS_DHM_PARAM_GY, /*!< The public key of the peer = \c G^Y mod \c P. */ - MBEDTLS_DHM_PARAM_K, /*!< The shared secret = \c G^(XY) mod \c P. */ -} mbedtls_dhm_parameter; - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_DHM_ALT) - -/** - * \brief The DHM context structure. - */ -typedef struct mbedtls_dhm_context -{ - mbedtls_mpi MBEDTLS_PRIVATE(P); /*!< The prime modulus. */ - mbedtls_mpi MBEDTLS_PRIVATE(G); /*!< The generator. */ - mbedtls_mpi MBEDTLS_PRIVATE(X); /*!< Our secret value. */ - mbedtls_mpi MBEDTLS_PRIVATE(GX); /*!< Our public key = \c G^X mod \c P. */ - mbedtls_mpi MBEDTLS_PRIVATE(GY); /*!< The public key of the peer = \c G^Y mod \c P. */ - mbedtls_mpi MBEDTLS_PRIVATE(K); /*!< The shared secret = \c G^(XY) mod \c P. */ - mbedtls_mpi MBEDTLS_PRIVATE(RP); /*!< The cached value = \c R^2 mod \c P. */ - mbedtls_mpi MBEDTLS_PRIVATE(Vi); /*!< The blinding value. */ - mbedtls_mpi MBEDTLS_PRIVATE(Vf); /*!< The unblinding value. */ - mbedtls_mpi MBEDTLS_PRIVATE(pX); /*!< The previous \c X. */ -} -mbedtls_dhm_context; - -#else /* MBEDTLS_DHM_ALT */ -#include "dhm_alt.h" -#endif /* MBEDTLS_DHM_ALT */ - -/** - * \brief This function initializes the DHM context. - * - * \param ctx The DHM context to initialize. - */ -void mbedtls_dhm_init( mbedtls_dhm_context *ctx ); - -/** - * \brief This function parses the DHM parameters in a - * TLS ServerKeyExchange handshake message - * (DHM modulus, generator, and public key). - * - * \note In a TLS handshake, this is the how the client - * sets up its DHM context from the server's public - * DHM key material. - * - * \param ctx The DHM context to use. This must be initialized. - * \param p On input, *p must be the start of the input buffer. - * On output, *p is updated to point to the end of the data - * that has been read. On success, this is the first byte - * past the end of the ServerKeyExchange parameters. - * On error, this is the point at which an error has been - * detected, which is usually not useful except to debug - * failures. - * \param end The end of the input buffer. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_DHM_XXX error code on failure. - */ -int mbedtls_dhm_read_params( mbedtls_dhm_context *ctx, - unsigned char **p, - const unsigned char *end ); - -/** - * \brief This function generates a DHM key pair and exports its - * public part together with the DHM parameters in the format - * used in a TLS ServerKeyExchange handshake message. - * - * \note This function assumes that the DHM parameters \c ctx->P - * and \c ctx->G have already been properly set. For that, use - * mbedtls_dhm_set_group() below in conjunction with - * mbedtls_mpi_read_binary() and mbedtls_mpi_read_string(). - * - * \note In a TLS handshake, this is the how the server generates - * and exports its DHM key material. - * - * \param ctx The DHM context to use. This must be initialized - * and have the DHM parameters set. It may or may not - * already have imported the peer's public key. - * \param x_size The private key size in Bytes. - * \param olen The address at which to store the number of Bytes - * written on success. This must not be \c NULL. - * \param output The destination buffer. This must be a writable buffer of - * sufficient size to hold the reduced binary presentation of - * the modulus, the generator and the public key, each wrapped - * with a 2-byte length field. It is the responsibility of the - * caller to ensure that enough space is available. Refer to - * mbedtls_mpi_size() to computing the byte-size of an MPI. - * \param f_rng The RNG function. Must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL if \p f_rng doesn't need a context parameter. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_DHM_XXX error code on failure. - */ -int mbedtls_dhm_make_params( mbedtls_dhm_context *ctx, int x_size, - unsigned char *output, size_t *olen, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function sets the prime modulus and generator. - * - * \note This function can be used to set \c ctx->P, \c ctx->G - * in preparation for mbedtls_dhm_make_params(). - * - * \param ctx The DHM context to configure. This must be initialized. - * \param P The MPI holding the DHM prime modulus. This must be - * an initialized MPI. - * \param G The MPI holding the DHM generator. This must be an - * initialized MPI. - * - * \return \c 0 if successful. - * \return An \c MBEDTLS_ERR_DHM_XXX error code on failure. - */ -int mbedtls_dhm_set_group( mbedtls_dhm_context *ctx, - const mbedtls_mpi *P, - const mbedtls_mpi *G ); - -/** - * \brief This function imports the raw public value of the peer. - * - * \note In a TLS handshake, this is the how the server imports - * the Client's public DHM key. - * - * \param ctx The DHM context to use. This must be initialized and have - * its DHM parameters set, e.g. via mbedtls_dhm_set_group(). - * It may or may not already have generated its own private key. - * \param input The input buffer containing the \c G^Y value of the peer. - * This must be a readable buffer of size \p ilen Bytes. - * \param ilen The size of the input buffer \p input in Bytes. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_DHM_XXX error code on failure. - */ -int mbedtls_dhm_read_public( mbedtls_dhm_context *ctx, - const unsigned char *input, size_t ilen ); - -/** - * \brief This function creates a DHM key pair and exports - * the raw public key in big-endian format. - * - * \note The destination buffer is always fully written - * so as to contain a big-endian representation of G^X mod P. - * If it is larger than \c ctx->len, it is padded accordingly - * with zero-bytes at the beginning. - * - * \param ctx The DHM context to use. This must be initialized and - * have the DHM parameters set. It may or may not already - * have imported the peer's public key. - * \param x_size The private key size in Bytes. - * \param output The destination buffer. This must be a writable buffer of - * size \p olen Bytes. - * \param olen The length of the destination buffer. This must be at least - * equal to `ctx->len` (the size of \c P). - * \param f_rng The RNG function. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be \c NULL - * if \p f_rng doesn't need a context argument. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_DHM_XXX error code on failure. - */ -int mbedtls_dhm_make_public( mbedtls_dhm_context *ctx, int x_size, - unsigned char *output, size_t olen, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function derives and exports the shared secret - * \c (G^Y)^X mod \c P. - * - * \note If \p f_rng is not \c NULL, it is used to blind the input as - * a countermeasure against timing attacks. Blinding is used - * only if our private key \c X is re-used, and not used - * otherwise. We recommend always passing a non-NULL - * \p f_rng argument. - * - * \param ctx The DHM context to use. This must be initialized - * and have its own private key generated and the peer's - * public key imported. - * \param output The buffer to write the generated shared key to. This - * must be a writable buffer of size \p output_size Bytes. - * \param output_size The size of the destination buffer. This must be at - * least the size of \c ctx->len (the size of \c P). - * \param olen On exit, holds the actual number of Bytes written. - * \param f_rng The RNG function. Must not be \c NULL. Used for - * blinding. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL if \p f_rng doesn't need a context parameter. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_DHM_XXX error code on failure. - */ -int mbedtls_dhm_calc_secret( mbedtls_dhm_context *ctx, - unsigned char *output, size_t output_size, size_t *olen, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function returns the size of the prime modulus in bits. - * - * \param ctx The DHM context to query. - * - * \return The size of the prime modulus in bits, - * i.e. the number n such that 2^(n-1) <= P < 2^n. - */ -size_t mbedtls_dhm_get_bitlen( const mbedtls_dhm_context *ctx ); - -/** - * \brief This function returns the size of the prime modulus in bytes. - * - * \param ctx The DHM context to query. - * - * \return The size of the prime modulus in bytes, - * i.e. the number n such that 2^(8*(n-1)) <= P < 2^(8*n). - */ -size_t mbedtls_dhm_get_len( const mbedtls_dhm_context *ctx ); - -/** - * \brief This function copies a parameter of a DHM key. - * - * \param ctx The DHM context to query. - * \param param The parameter to copy. - * \param dest The MPI object to copy the value into. It must be - * initialized. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_DHM_BAD_INPUT_DATA if \p field is invalid. - * \return An \c MBEDTLS_ERR_MPI_XXX error code if the copy fails. - */ -int mbedtls_dhm_get_value( const mbedtls_dhm_context *ctx, - mbedtls_dhm_parameter param, - mbedtls_mpi *dest ); - -/** - * \brief This function frees and clears the components - * of a DHM context. - * - * \param ctx The DHM context to free and clear. This may be \c NULL, - * in which case this function is a no-op. If it is not \c NULL, - * it must point to an initialized DHM context. - */ -void mbedtls_dhm_free( mbedtls_dhm_context *ctx ); - -#if defined(MBEDTLS_ASN1_PARSE_C) -/** - * \brief This function parses DHM parameters in PEM or DER format. - * - * \param dhm The DHM context to import the DHM parameters into. - * This must be initialized. - * \param dhmin The input buffer. This must be a readable buffer of - * length \p dhminlen Bytes. - * \param dhminlen The size of the input buffer \p dhmin, including the - * terminating \c NULL Byte for PEM data. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_DHM_XXX or \c MBEDTLS_ERR_PEM_XXX error - * code on failure. - */ -int mbedtls_dhm_parse_dhm( mbedtls_dhm_context *dhm, const unsigned char *dhmin, - size_t dhminlen ); - -#if defined(MBEDTLS_FS_IO) -/** - * \brief This function loads and parses DHM parameters from a file. - * - * \param dhm The DHM context to load the parameters to. - * This must be initialized. - * \param path The filename to read the DHM parameters from. - * This must not be \c NULL. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_DHM_XXX or \c MBEDTLS_ERR_PEM_XXX - * error code on failure. - */ -int mbedtls_dhm_parse_dhmfile( mbedtls_dhm_context *dhm, const char *path ); -#endif /* MBEDTLS_FS_IO */ -#endif /* MBEDTLS_ASN1_PARSE_C */ - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief The DMH checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_dhm_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ -#ifdef __cplusplus -} -#endif - -/** - * RFC 3526, RFC 5114 and RFC 7919 standardize a number of - * Diffie-Hellman groups, some of which are included here - * for use within the SSL/TLS module and the user's convenience - * when configuring the Diffie-Hellman parameters by hand - * through \c mbedtls_ssl_conf_dh_param. - * - * The following lists the source of the above groups in the standards: - * - RFC 5114 section 2.2: 2048-bit MODP Group with 224-bit Prime Order Subgroup - * - RFC 3526 section 3: 2048-bit MODP Group - * - RFC 3526 section 4: 3072-bit MODP Group - * - RFC 3526 section 5: 4096-bit MODP Group - * - RFC 7919 section A.1: ffdhe2048 - * - RFC 7919 section A.2: ffdhe3072 - * - RFC 7919 section A.3: ffdhe4096 - * - RFC 7919 section A.4: ffdhe6144 - * - RFC 7919 section A.5: ffdhe8192 - * - * The constants with suffix "_p" denote the chosen prime moduli, while - * the constants with suffix "_g" denote the chosen generator - * of the associated prime field. - * - * The constants further suffixed with "_bin" are provided in binary format, - * while all other constants represent null-terminated strings holding the - * hexadecimal presentation of the respective numbers. - * - * The primes from RFC 3526 and RFC 7919 have been generating by the following - * trust-worthy procedure: - * - Fix N in { 2048, 3072, 4096, 6144, 8192 } and consider the N-bit number - * the first and last 64 bits are all 1, and the remaining N - 128 bits of - * which are 0x7ff...ff. - * - Add the smallest multiple of the first N - 129 bits of the binary expansion - * of pi (for RFC 5236) or e (for RFC 7919) to this intermediate bit-string - * such that the resulting integer is a safe-prime. - * - The result is the respective RFC 3526 / 7919 prime, and the corresponding - * generator is always chosen to be 2 (which is a square for these prime, - * hence the corresponding subgroup has order (p-1)/2 and avoids leaking a - * bit in the private exponent). - * - */ - -/* - * Trustworthy DHM parameters in binary form - */ - -#define MBEDTLS_DHM_RFC3526_MODP_2048_P_BIN { \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, \ - 0xC9, 0x0F, 0xDA, 0xA2, 0x21, 0x68, 0xC2, 0x34, \ - 0xC4, 0xC6, 0x62, 0x8B, 0x80, 0xDC, 0x1C, 0xD1, \ - 0x29, 0x02, 0x4E, 0x08, 0x8A, 0x67, 0xCC, 0x74, \ - 0x02, 0x0B, 0xBE, 0xA6, 0x3B, 0x13, 0x9B, 0x22, \ - 0x51, 0x4A, 0x08, 0x79, 0x8E, 0x34, 0x04, 0xDD, \ - 0xEF, 0x95, 0x19, 0xB3, 0xCD, 0x3A, 0x43, 0x1B, \ - 0x30, 0x2B, 0x0A, 0x6D, 0xF2, 0x5F, 0x14, 0x37, \ - 0x4F, 0xE1, 0x35, 0x6D, 0x6D, 0x51, 0xC2, 0x45, \ - 0xE4, 0x85, 0xB5, 0x76, 0x62, 0x5E, 0x7E, 0xC6, \ - 0xF4, 0x4C, 0x42, 0xE9, 0xA6, 0x37, 0xED, 0x6B, \ - 0x0B, 0xFF, 0x5C, 0xB6, 0xF4, 0x06, 0xB7, 0xED, \ - 0xEE, 0x38, 0x6B, 0xFB, 0x5A, 0x89, 0x9F, 0xA5, \ - 0xAE, 0x9F, 0x24, 0x11, 0x7C, 0x4B, 0x1F, 0xE6, \ - 0x49, 0x28, 0x66, 0x51, 0xEC, 0xE4, 0x5B, 0x3D, \ - 0xC2, 0x00, 0x7C, 0xB8, 0xA1, 0x63, 0xBF, 0x05, \ - 0x98, 0xDA, 0x48, 0x36, 0x1C, 0x55, 0xD3, 0x9A, \ - 0x69, 0x16, 0x3F, 0xA8, 0xFD, 0x24, 0xCF, 0x5F, \ - 0x83, 0x65, 0x5D, 0x23, 0xDC, 0xA3, 0xAD, 0x96, \ - 0x1C, 0x62, 0xF3, 0x56, 0x20, 0x85, 0x52, 0xBB, \ - 0x9E, 0xD5, 0x29, 0x07, 0x70, 0x96, 0x96, 0x6D, \ - 0x67, 0x0C, 0x35, 0x4E, 0x4A, 0xBC, 0x98, 0x04, \ - 0xF1, 0x74, 0x6C, 0x08, 0xCA, 0x18, 0x21, 0x7C, \ - 0x32, 0x90, 0x5E, 0x46, 0x2E, 0x36, 0xCE, 0x3B, \ - 0xE3, 0x9E, 0x77, 0x2C, 0x18, 0x0E, 0x86, 0x03, \ - 0x9B, 0x27, 0x83, 0xA2, 0xEC, 0x07, 0xA2, 0x8F, \ - 0xB5, 0xC5, 0x5D, 0xF0, 0x6F, 0x4C, 0x52, 0xC9, \ - 0xDE, 0x2B, 0xCB, 0xF6, 0x95, 0x58, 0x17, 0x18, \ - 0x39, 0x95, 0x49, 0x7C, 0xEA, 0x95, 0x6A, 0xE5, \ - 0x15, 0xD2, 0x26, 0x18, 0x98, 0xFA, 0x05, 0x10, \ - 0x15, 0x72, 0x8E, 0x5A, 0x8A, 0xAC, 0xAA, 0x68, \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF } - -#define MBEDTLS_DHM_RFC3526_MODP_2048_G_BIN { 0x02 } - -#define MBEDTLS_DHM_RFC3526_MODP_3072_P_BIN { \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, \ - 0xC9, 0x0F, 0xDA, 0xA2, 0x21, 0x68, 0xC2, 0x34, \ - 0xC4, 0xC6, 0x62, 0x8B, 0x80, 0xDC, 0x1C, 0xD1, \ - 0x29, 0x02, 0x4E, 0x08, 0x8A, 0x67, 0xCC, 0x74, \ - 0x02, 0x0B, 0xBE, 0xA6, 0x3B, 0x13, 0x9B, 0x22, \ - 0x51, 0x4A, 0x08, 0x79, 0x8E, 0x34, 0x04, 0xDD, \ - 0xEF, 0x95, 0x19, 0xB3, 0xCD, 0x3A, 0x43, 0x1B, \ - 0x30, 0x2B, 0x0A, 0x6D, 0xF2, 0x5F, 0x14, 0x37, \ - 0x4F, 0xE1, 0x35, 0x6D, 0x6D, 0x51, 0xC2, 0x45, \ - 0xE4, 0x85, 0xB5, 0x76, 0x62, 0x5E, 0x7E, 0xC6, \ - 0xF4, 0x4C, 0x42, 0xE9, 0xA6, 0x37, 0xED, 0x6B, \ - 0x0B, 0xFF, 0x5C, 0xB6, 0xF4, 0x06, 0xB7, 0xED, \ - 0xEE, 0x38, 0x6B, 0xFB, 0x5A, 0x89, 0x9F, 0xA5, \ - 0xAE, 0x9F, 0x24, 0x11, 0x7C, 0x4B, 0x1F, 0xE6, \ - 0x49, 0x28, 0x66, 0x51, 0xEC, 0xE4, 0x5B, 0x3D, \ - 0xC2, 0x00, 0x7C, 0xB8, 0xA1, 0x63, 0xBF, 0x05, \ - 0x98, 0xDA, 0x48, 0x36, 0x1C, 0x55, 0xD3, 0x9A, \ - 0x69, 0x16, 0x3F, 0xA8, 0xFD, 0x24, 0xCF, 0x5F, \ - 0x83, 0x65, 0x5D, 0x23, 0xDC, 0xA3, 0xAD, 0x96, \ - 0x1C, 0x62, 0xF3, 0x56, 0x20, 0x85, 0x52, 0xBB, \ - 0x9E, 0xD5, 0x29, 0x07, 0x70, 0x96, 0x96, 0x6D, \ - 0x67, 0x0C, 0x35, 0x4E, 0x4A, 0xBC, 0x98, 0x04, \ - 0xF1, 0x74, 0x6C, 0x08, 0xCA, 0x18, 0x21, 0x7C, \ - 0x32, 0x90, 0x5E, 0x46, 0x2E, 0x36, 0xCE, 0x3B, \ - 0xE3, 0x9E, 0x77, 0x2C, 0x18, 0x0E, 0x86, 0x03, \ - 0x9B, 0x27, 0x83, 0xA2, 0xEC, 0x07, 0xA2, 0x8F, \ - 0xB5, 0xC5, 0x5D, 0xF0, 0x6F, 0x4C, 0x52, 0xC9, \ - 0xDE, 0x2B, 0xCB, 0xF6, 0x95, 0x58, 0x17, 0x18, \ - 0x39, 0x95, 0x49, 0x7C, 0xEA, 0x95, 0x6A, 0xE5, \ - 0x15, 0xD2, 0x26, 0x18, 0x98, 0xFA, 0x05, 0x10, \ - 0x15, 0x72, 0x8E, 0x5A, 0x8A, 0xAA, 0xC4, 0x2D, \ - 0xAD, 0x33, 0x17, 0x0D, 0x04, 0x50, 0x7A, 0x33, \ - 0xA8, 0x55, 0x21, 0xAB, 0xDF, 0x1C, 0xBA, 0x64, \ - 0xEC, 0xFB, 0x85, 0x04, 0x58, 0xDB, 0xEF, 0x0A, \ - 0x8A, 0xEA, 0x71, 0x57, 0x5D, 0x06, 0x0C, 0x7D, \ - 0xB3, 0x97, 0x0F, 0x85, 0xA6, 0xE1, 0xE4, 0xC7, \ - 0xAB, 0xF5, 0xAE, 0x8C, 0xDB, 0x09, 0x33, 0xD7, \ - 0x1E, 0x8C, 0x94, 0xE0, 0x4A, 0x25, 0x61, 0x9D, \ - 0xCE, 0xE3, 0xD2, 0x26, 0x1A, 0xD2, 0xEE, 0x6B, \ - 0xF1, 0x2F, 0xFA, 0x06, 0xD9, 0x8A, 0x08, 0x64, \ - 0xD8, 0x76, 0x02, 0x73, 0x3E, 0xC8, 0x6A, 0x64, \ - 0x52, 0x1F, 0x2B, 0x18, 0x17, 0x7B, 0x20, 0x0C, \ - 0xBB, 0xE1, 0x17, 0x57, 0x7A, 0x61, 0x5D, 0x6C, \ - 0x77, 0x09, 0x88, 0xC0, 0xBA, 0xD9, 0x46, 0xE2, \ - 0x08, 0xE2, 0x4F, 0xA0, 0x74, 0xE5, 0xAB, 0x31, \ - 0x43, 0xDB, 0x5B, 0xFC, 0xE0, 0xFD, 0x10, 0x8E, \ - 0x4B, 0x82, 0xD1, 0x20, 0xA9, 0x3A, 0xD2, 0xCA, \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF } - -#define MBEDTLS_DHM_RFC3526_MODP_3072_G_BIN { 0x02 } - -#define MBEDTLS_DHM_RFC3526_MODP_4096_P_BIN { \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, \ - 0xC9, 0x0F, 0xDA, 0xA2, 0x21, 0x68, 0xC2, 0x34, \ - 0xC4, 0xC6, 0x62, 0x8B, 0x80, 0xDC, 0x1C, 0xD1, \ - 0x29, 0x02, 0x4E, 0x08, 0x8A, 0x67, 0xCC, 0x74, \ - 0x02, 0x0B, 0xBE, 0xA6, 0x3B, 0x13, 0x9B, 0x22, \ - 0x51, 0x4A, 0x08, 0x79, 0x8E, 0x34, 0x04, 0xDD, \ - 0xEF, 0x95, 0x19, 0xB3, 0xCD, 0x3A, 0x43, 0x1B, \ - 0x30, 0x2B, 0x0A, 0x6D, 0xF2, 0x5F, 0x14, 0x37, \ - 0x4F, 0xE1, 0x35, 0x6D, 0x6D, 0x51, 0xC2, 0x45, \ - 0xE4, 0x85, 0xB5, 0x76, 0x62, 0x5E, 0x7E, 0xC6, \ - 0xF4, 0x4C, 0x42, 0xE9, 0xA6, 0x37, 0xED, 0x6B, \ - 0x0B, 0xFF, 0x5C, 0xB6, 0xF4, 0x06, 0xB7, 0xED, \ - 0xEE, 0x38, 0x6B, 0xFB, 0x5A, 0x89, 0x9F, 0xA5, \ - 0xAE, 0x9F, 0x24, 0x11, 0x7C, 0x4B, 0x1F, 0xE6, \ - 0x49, 0x28, 0x66, 0x51, 0xEC, 0xE4, 0x5B, 0x3D, \ - 0xC2, 0x00, 0x7C, 0xB8, 0xA1, 0x63, 0xBF, 0x05, \ - 0x98, 0xDA, 0x48, 0x36, 0x1C, 0x55, 0xD3, 0x9A, \ - 0x69, 0x16, 0x3F, 0xA8, 0xFD, 0x24, 0xCF, 0x5F, \ - 0x83, 0x65, 0x5D, 0x23, 0xDC, 0xA3, 0xAD, 0x96, \ - 0x1C, 0x62, 0xF3, 0x56, 0x20, 0x85, 0x52, 0xBB, \ - 0x9E, 0xD5, 0x29, 0x07, 0x70, 0x96, 0x96, 0x6D, \ - 0x67, 0x0C, 0x35, 0x4E, 0x4A, 0xBC, 0x98, 0x04, \ - 0xF1, 0x74, 0x6C, 0x08, 0xCA, 0x18, 0x21, 0x7C, \ - 0x32, 0x90, 0x5E, 0x46, 0x2E, 0x36, 0xCE, 0x3B, \ - 0xE3, 0x9E, 0x77, 0x2C, 0x18, 0x0E, 0x86, 0x03, \ - 0x9B, 0x27, 0x83, 0xA2, 0xEC, 0x07, 0xA2, 0x8F, \ - 0xB5, 0xC5, 0x5D, 0xF0, 0x6F, 0x4C, 0x52, 0xC9, \ - 0xDE, 0x2B, 0xCB, 0xF6, 0x95, 0x58, 0x17, 0x18, \ - 0x39, 0x95, 0x49, 0x7C, 0xEA, 0x95, 0x6A, 0xE5, \ - 0x15, 0xD2, 0x26, 0x18, 0x98, 0xFA, 0x05, 0x10, \ - 0x15, 0x72, 0x8E, 0x5A, 0x8A, 0xAA, 0xC4, 0x2D, \ - 0xAD, 0x33, 0x17, 0x0D, 0x04, 0x50, 0x7A, 0x33, \ - 0xA8, 0x55, 0x21, 0xAB, 0xDF, 0x1C, 0xBA, 0x64, \ - 0xEC, 0xFB, 0x85, 0x04, 0x58, 0xDB, 0xEF, 0x0A, \ - 0x8A, 0xEA, 0x71, 0x57, 0x5D, 0x06, 0x0C, 0x7D, \ - 0xB3, 0x97, 0x0F, 0x85, 0xA6, 0xE1, 0xE4, 0xC7, \ - 0xAB, 0xF5, 0xAE, 0x8C, 0xDB, 0x09, 0x33, 0xD7, \ - 0x1E, 0x8C, 0x94, 0xE0, 0x4A, 0x25, 0x61, 0x9D, \ - 0xCE, 0xE3, 0xD2, 0x26, 0x1A, 0xD2, 0xEE, 0x6B, \ - 0xF1, 0x2F, 0xFA, 0x06, 0xD9, 0x8A, 0x08, 0x64, \ - 0xD8, 0x76, 0x02, 0x73, 0x3E, 0xC8, 0x6A, 0x64, \ - 0x52, 0x1F, 0x2B, 0x18, 0x17, 0x7B, 0x20, 0x0C, \ - 0xBB, 0xE1, 0x17, 0x57, 0x7A, 0x61, 0x5D, 0x6C, \ - 0x77, 0x09, 0x88, 0xC0, 0xBA, 0xD9, 0x46, 0xE2, \ - 0x08, 0xE2, 0x4F, 0xA0, 0x74, 0xE5, 0xAB, 0x31, \ - 0x43, 0xDB, 0x5B, 0xFC, 0xE0, 0xFD, 0x10, 0x8E, \ - 0x4B, 0x82, 0xD1, 0x20, 0xA9, 0x21, 0x08, 0x01, \ - 0x1A, 0x72, 0x3C, 0x12, 0xA7, 0x87, 0xE6, 0xD7, \ - 0x88, 0x71, 0x9A, 0x10, 0xBD, 0xBA, 0x5B, 0x26, \ - 0x99, 0xC3, 0x27, 0x18, 0x6A, 0xF4, 0xE2, 0x3C, \ - 0x1A, 0x94, 0x68, 0x34, 0xB6, 0x15, 0x0B, 0xDA, \ - 0x25, 0x83, 0xE9, 0xCA, 0x2A, 0xD4, 0x4C, 0xE8, \ - 0xDB, 0xBB, 0xC2, 0xDB, 0x04, 0xDE, 0x8E, 0xF9, \ - 0x2E, 0x8E, 0xFC, 0x14, 0x1F, 0xBE, 0xCA, 0xA6, \ - 0x28, 0x7C, 0x59, 0x47, 0x4E, 0x6B, 0xC0, 0x5D, \ - 0x99, 0xB2, 0x96, 0x4F, 0xA0, 0x90, 0xC3, 0xA2, \ - 0x23, 0x3B, 0xA1, 0x86, 0x51, 0x5B, 0xE7, 0xED, \ - 0x1F, 0x61, 0x29, 0x70, 0xCE, 0xE2, 0xD7, 0xAF, \ - 0xB8, 0x1B, 0xDD, 0x76, 0x21, 0x70, 0x48, 0x1C, \ - 0xD0, 0x06, 0x91, 0x27, 0xD5, 0xB0, 0x5A, 0xA9, \ - 0x93, 0xB4, 0xEA, 0x98, 0x8D, 0x8F, 0xDD, 0xC1, \ - 0x86, 0xFF, 0xB7, 0xDC, 0x90, 0xA6, 0xC0, 0x8F, \ - 0x4D, 0xF4, 0x35, 0xC9, 0x34, 0x06, 0x31, 0x99, \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF } - -#define MBEDTLS_DHM_RFC3526_MODP_4096_G_BIN { 0x02 } - -#define MBEDTLS_DHM_RFC7919_FFDHE2048_P_BIN { \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, \ - 0xAD, 0xF8, 0x54, 0x58, 0xA2, 0xBB, 0x4A, 0x9A, \ - 0xAF, 0xDC, 0x56, 0x20, 0x27, 0x3D, 0x3C, 0xF1, \ - 0xD8, 0xB9, 0xC5, 0x83, 0xCE, 0x2D, 0x36, 0x95, \ - 0xA9, 0xE1, 0x36, 0x41, 0x14, 0x64, 0x33, 0xFB, \ - 0xCC, 0x93, 0x9D, 0xCE, 0x24, 0x9B, 0x3E, 0xF9, \ - 0x7D, 0x2F, 0xE3, 0x63, 0x63, 0x0C, 0x75, 0xD8, \ - 0xF6, 0x81, 0xB2, 0x02, 0xAE, 0xC4, 0x61, 0x7A, \ - 0xD3, 0xDF, 0x1E, 0xD5, 0xD5, 0xFD, 0x65, 0x61, \ - 0x24, 0x33, 0xF5, 0x1F, 0x5F, 0x06, 0x6E, 0xD0, \ - 0x85, 0x63, 0x65, 0x55, 0x3D, 0xED, 0x1A, 0xF3, \ - 0xB5, 0x57, 0x13, 0x5E, 0x7F, 0x57, 0xC9, 0x35, \ - 0x98, 0x4F, 0x0C, 0x70, 0xE0, 0xE6, 0x8B, 0x77, \ - 0xE2, 0xA6, 0x89, 0xDA, 0xF3, 0xEF, 0xE8, 0x72, \ - 0x1D, 0xF1, 0x58, 0xA1, 0x36, 0xAD, 0xE7, 0x35, \ - 0x30, 0xAC, 0xCA, 0x4F, 0x48, 0x3A, 0x79, 0x7A, \ - 0xBC, 0x0A, 0xB1, 0x82, 0xB3, 0x24, 0xFB, 0x61, \ - 0xD1, 0x08, 0xA9, 0x4B, 0xB2, 0xC8, 0xE3, 0xFB, \ - 0xB9, 0x6A, 0xDA, 0xB7, 0x60, 0xD7, 0xF4, 0x68, \ - 0x1D, 0x4F, 0x42, 0xA3, 0xDE, 0x39, 0x4D, 0xF4, \ - 0xAE, 0x56, 0xED, 0xE7, 0x63, 0x72, 0xBB, 0x19, \ - 0x0B, 0x07, 0xA7, 0xC8, 0xEE, 0x0A, 0x6D, 0x70, \ - 0x9E, 0x02, 0xFC, 0xE1, 0xCD, 0xF7, 0xE2, 0xEC, \ - 0xC0, 0x34, 0x04, 0xCD, 0x28, 0x34, 0x2F, 0x61, \ - 0x91, 0x72, 0xFE, 0x9C, 0xE9, 0x85, 0x83, 0xFF, \ - 0x8E, 0x4F, 0x12, 0x32, 0xEE, 0xF2, 0x81, 0x83, \ - 0xC3, 0xFE, 0x3B, 0x1B, 0x4C, 0x6F, 0xAD, 0x73, \ - 0x3B, 0xB5, 0xFC, 0xBC, 0x2E, 0xC2, 0x20, 0x05, \ - 0xC5, 0x8E, 0xF1, 0x83, 0x7D, 0x16, 0x83, 0xB2, \ - 0xC6, 0xF3, 0x4A, 0x26, 0xC1, 0xB2, 0xEF, 0xFA, \ - 0x88, 0x6B, 0x42, 0x38, 0x61, 0x28, 0x5C, 0x97, \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, } - -#define MBEDTLS_DHM_RFC7919_FFDHE2048_G_BIN { 0x02 } - -#define MBEDTLS_DHM_RFC7919_FFDHE3072_P_BIN { \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, \ - 0xAD, 0xF8, 0x54, 0x58, 0xA2, 0xBB, 0x4A, 0x9A, \ - 0xAF, 0xDC, 0x56, 0x20, 0x27, 0x3D, 0x3C, 0xF1, \ - 0xD8, 0xB9, 0xC5, 0x83, 0xCE, 0x2D, 0x36, 0x95, \ - 0xA9, 0xE1, 0x36, 0x41, 0x14, 0x64, 0x33, 0xFB, \ - 0xCC, 0x93, 0x9D, 0xCE, 0x24, 0x9B, 0x3E, 0xF9, \ - 0x7D, 0x2F, 0xE3, 0x63, 0x63, 0x0C, 0x75, 0xD8, \ - 0xF6, 0x81, 0xB2, 0x02, 0xAE, 0xC4, 0x61, 0x7A, \ - 0xD3, 0xDF, 0x1E, 0xD5, 0xD5, 0xFD, 0x65, 0x61, \ - 0x24, 0x33, 0xF5, 0x1F, 0x5F, 0x06, 0x6E, 0xD0, \ - 0x85, 0x63, 0x65, 0x55, 0x3D, 0xED, 0x1A, 0xF3, \ - 0xB5, 0x57, 0x13, 0x5E, 0x7F, 0x57, 0xC9, 0x35, \ - 0x98, 0x4F, 0x0C, 0x70, 0xE0, 0xE6, 0x8B, 0x77, \ - 0xE2, 0xA6, 0x89, 0xDA, 0xF3, 0xEF, 0xE8, 0x72, \ - 0x1D, 0xF1, 0x58, 0xA1, 0x36, 0xAD, 0xE7, 0x35, \ - 0x30, 0xAC, 0xCA, 0x4F, 0x48, 0x3A, 0x79, 0x7A, \ - 0xBC, 0x0A, 0xB1, 0x82, 0xB3, 0x24, 0xFB, 0x61, \ - 0xD1, 0x08, 0xA9, 0x4B, 0xB2, 0xC8, 0xE3, 0xFB, \ - 0xB9, 0x6A, 0xDA, 0xB7, 0x60, 0xD7, 0xF4, 0x68, \ - 0x1D, 0x4F, 0x42, 0xA3, 0xDE, 0x39, 0x4D, 0xF4, \ - 0xAE, 0x56, 0xED, 0xE7, 0x63, 0x72, 0xBB, 0x19, \ - 0x0B, 0x07, 0xA7, 0xC8, 0xEE, 0x0A, 0x6D, 0x70, \ - 0x9E, 0x02, 0xFC, 0xE1, 0xCD, 0xF7, 0xE2, 0xEC, \ - 0xC0, 0x34, 0x04, 0xCD, 0x28, 0x34, 0x2F, 0x61, \ - 0x91, 0x72, 0xFE, 0x9C, 0xE9, 0x85, 0x83, 0xFF, \ - 0x8E, 0x4F, 0x12, 0x32, 0xEE, 0xF2, 0x81, 0x83, \ - 0xC3, 0xFE, 0x3B, 0x1B, 0x4C, 0x6F, 0xAD, 0x73, \ - 0x3B, 0xB5, 0xFC, 0xBC, 0x2E, 0xC2, 0x20, 0x05, \ - 0xC5, 0x8E, 0xF1, 0x83, 0x7D, 0x16, 0x83, 0xB2, \ - 0xC6, 0xF3, 0x4A, 0x26, 0xC1, 0xB2, 0xEF, 0xFA, \ - 0x88, 0x6B, 0x42, 0x38, 0x61, 0x1F, 0xCF, 0xDC, \ - 0xDE, 0x35, 0x5B, 0x3B, 0x65, 0x19, 0x03, 0x5B, \ - 0xBC, 0x34, 0xF4, 0xDE, 0xF9, 0x9C, 0x02, 0x38, \ - 0x61, 0xB4, 0x6F, 0xC9, 0xD6, 0xE6, 0xC9, 0x07, \ - 0x7A, 0xD9, 0x1D, 0x26, 0x91, 0xF7, 0xF7, 0xEE, \ - 0x59, 0x8C, 0xB0, 0xFA, 0xC1, 0x86, 0xD9, 0x1C, \ - 0xAE, 0xFE, 0x13, 0x09, 0x85, 0x13, 0x92, 0x70, \ - 0xB4, 0x13, 0x0C, 0x93, 0xBC, 0x43, 0x79, 0x44, \ - 0xF4, 0xFD, 0x44, 0x52, 0xE2, 0xD7, 0x4D, 0xD3, \ - 0x64, 0xF2, 0xE2, 0x1E, 0x71, 0xF5, 0x4B, 0xFF, \ - 0x5C, 0xAE, 0x82, 0xAB, 0x9C, 0x9D, 0xF6, 0x9E, \ - 0xE8, 0x6D, 0x2B, 0xC5, 0x22, 0x36, 0x3A, 0x0D, \ - 0xAB, 0xC5, 0x21, 0x97, 0x9B, 0x0D, 0xEA, 0xDA, \ - 0x1D, 0xBF, 0x9A, 0x42, 0xD5, 0xC4, 0x48, 0x4E, \ - 0x0A, 0xBC, 0xD0, 0x6B, 0xFA, 0x53, 0xDD, 0xEF, \ - 0x3C, 0x1B, 0x20, 0xEE, 0x3F, 0xD5, 0x9D, 0x7C, \ - 0x25, 0xE4, 0x1D, 0x2B, 0x66, 0xC6, 0x2E, 0x37, \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF } - -#define MBEDTLS_DHM_RFC7919_FFDHE3072_G_BIN { 0x02 } - -#define MBEDTLS_DHM_RFC7919_FFDHE4096_P_BIN { \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, \ - 0xAD, 0xF8, 0x54, 0x58, 0xA2, 0xBB, 0x4A, 0x9A, \ - 0xAF, 0xDC, 0x56, 0x20, 0x27, 0x3D, 0x3C, 0xF1, \ - 0xD8, 0xB9, 0xC5, 0x83, 0xCE, 0x2D, 0x36, 0x95, \ - 0xA9, 0xE1, 0x36, 0x41, 0x14, 0x64, 0x33, 0xFB, \ - 0xCC, 0x93, 0x9D, 0xCE, 0x24, 0x9B, 0x3E, 0xF9, \ - 0x7D, 0x2F, 0xE3, 0x63, 0x63, 0x0C, 0x75, 0xD8, \ - 0xF6, 0x81, 0xB2, 0x02, 0xAE, 0xC4, 0x61, 0x7A, \ - 0xD3, 0xDF, 0x1E, 0xD5, 0xD5, 0xFD, 0x65, 0x61, \ - 0x24, 0x33, 0xF5, 0x1F, 0x5F, 0x06, 0x6E, 0xD0, \ - 0x85, 0x63, 0x65, 0x55, 0x3D, 0xED, 0x1A, 0xF3, \ - 0xB5, 0x57, 0x13, 0x5E, 0x7F, 0x57, 0xC9, 0x35, \ - 0x98, 0x4F, 0x0C, 0x70, 0xE0, 0xE6, 0x8B, 0x77, \ - 0xE2, 0xA6, 0x89, 0xDA, 0xF3, 0xEF, 0xE8, 0x72, \ - 0x1D, 0xF1, 0x58, 0xA1, 0x36, 0xAD, 0xE7, 0x35, \ - 0x30, 0xAC, 0xCA, 0x4F, 0x48, 0x3A, 0x79, 0x7A, \ - 0xBC, 0x0A, 0xB1, 0x82, 0xB3, 0x24, 0xFB, 0x61, \ - 0xD1, 0x08, 0xA9, 0x4B, 0xB2, 0xC8, 0xE3, 0xFB, \ - 0xB9, 0x6A, 0xDA, 0xB7, 0x60, 0xD7, 0xF4, 0x68, \ - 0x1D, 0x4F, 0x42, 0xA3, 0xDE, 0x39, 0x4D, 0xF4, \ - 0xAE, 0x56, 0xED, 0xE7, 0x63, 0x72, 0xBB, 0x19, \ - 0x0B, 0x07, 0xA7, 0xC8, 0xEE, 0x0A, 0x6D, 0x70, \ - 0x9E, 0x02, 0xFC, 0xE1, 0xCD, 0xF7, 0xE2, 0xEC, \ - 0xC0, 0x34, 0x04, 0xCD, 0x28, 0x34, 0x2F, 0x61, \ - 0x91, 0x72, 0xFE, 0x9C, 0xE9, 0x85, 0x83, 0xFF, \ - 0x8E, 0x4F, 0x12, 0x32, 0xEE, 0xF2, 0x81, 0x83, \ - 0xC3, 0xFE, 0x3B, 0x1B, 0x4C, 0x6F, 0xAD, 0x73, \ - 0x3B, 0xB5, 0xFC, 0xBC, 0x2E, 0xC2, 0x20, 0x05, \ - 0xC5, 0x8E, 0xF1, 0x83, 0x7D, 0x16, 0x83, 0xB2, \ - 0xC6, 0xF3, 0x4A, 0x26, 0xC1, 0xB2, 0xEF, 0xFA, \ - 0x88, 0x6B, 0x42, 0x38, 0x61, 0x1F, 0xCF, 0xDC, \ - 0xDE, 0x35, 0x5B, 0x3B, 0x65, 0x19, 0x03, 0x5B, \ - 0xBC, 0x34, 0xF4, 0xDE, 0xF9, 0x9C, 0x02, 0x38, \ - 0x61, 0xB4, 0x6F, 0xC9, 0xD6, 0xE6, 0xC9, 0x07, \ - 0x7A, 0xD9, 0x1D, 0x26, 0x91, 0xF7, 0xF7, 0xEE, \ - 0x59, 0x8C, 0xB0, 0xFA, 0xC1, 0x86, 0xD9, 0x1C, \ - 0xAE, 0xFE, 0x13, 0x09, 0x85, 0x13, 0x92, 0x70, \ - 0xB4, 0x13, 0x0C, 0x93, 0xBC, 0x43, 0x79, 0x44, \ - 0xF4, 0xFD, 0x44, 0x52, 0xE2, 0xD7, 0x4D, 0xD3, \ - 0x64, 0xF2, 0xE2, 0x1E, 0x71, 0xF5, 0x4B, 0xFF, \ - 0x5C, 0xAE, 0x82, 0xAB, 0x9C, 0x9D, 0xF6, 0x9E, \ - 0xE8, 0x6D, 0x2B, 0xC5, 0x22, 0x36, 0x3A, 0x0D, \ - 0xAB, 0xC5, 0x21, 0x97, 0x9B, 0x0D, 0xEA, 0xDA, \ - 0x1D, 0xBF, 0x9A, 0x42, 0xD5, 0xC4, 0x48, 0x4E, \ - 0x0A, 0xBC, 0xD0, 0x6B, 0xFA, 0x53, 0xDD, 0xEF, \ - 0x3C, 0x1B, 0x20, 0xEE, 0x3F, 0xD5, 0x9D, 0x7C, \ - 0x25, 0xE4, 0x1D, 0x2B, 0x66, 0x9E, 0x1E, 0xF1, \ - 0x6E, 0x6F, 0x52, 0xC3, 0x16, 0x4D, 0xF4, 0xFB, \ - 0x79, 0x30, 0xE9, 0xE4, 0xE5, 0x88, 0x57, 0xB6, \ - 0xAC, 0x7D, 0x5F, 0x42, 0xD6, 0x9F, 0x6D, 0x18, \ - 0x77, 0x63, 0xCF, 0x1D, 0x55, 0x03, 0x40, 0x04, \ - 0x87, 0xF5, 0x5B, 0xA5, 0x7E, 0x31, 0xCC, 0x7A, \ - 0x71, 0x35, 0xC8, 0x86, 0xEF, 0xB4, 0x31, 0x8A, \ - 0xED, 0x6A, 0x1E, 0x01, 0x2D, 0x9E, 0x68, 0x32, \ - 0xA9, 0x07, 0x60, 0x0A, 0x91, 0x81, 0x30, 0xC4, \ - 0x6D, 0xC7, 0x78, 0xF9, 0x71, 0xAD, 0x00, 0x38, \ - 0x09, 0x29, 0x99, 0xA3, 0x33, 0xCB, 0x8B, 0x7A, \ - 0x1A, 0x1D, 0xB9, 0x3D, 0x71, 0x40, 0x00, 0x3C, \ - 0x2A, 0x4E, 0xCE, 0xA9, 0xF9, 0x8D, 0x0A, 0xCC, \ - 0x0A, 0x82, 0x91, 0xCD, 0xCE, 0xC9, 0x7D, 0xCF, \ - 0x8E, 0xC9, 0xB5, 0x5A, 0x7F, 0x88, 0xA4, 0x6B, \ - 0x4D, 0xB5, 0xA8, 0x51, 0xF4, 0x41, 0x82, 0xE1, \ - 0xC6, 0x8A, 0x00, 0x7E, 0x5E, 0x65, 0x5F, 0x6A, \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF } - -#define MBEDTLS_DHM_RFC7919_FFDHE4096_G_BIN { 0x02 } - -#define MBEDTLS_DHM_RFC7919_FFDHE6144_P_BIN { \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, \ - 0xAD, 0xF8, 0x54, 0x58, 0xA2, 0xBB, 0x4A, 0x9A, \ - 0xAF, 0xDC, 0x56, 0x20, 0x27, 0x3D, 0x3C, 0xF1, \ - 0xD8, 0xB9, 0xC5, 0x83, 0xCE, 0x2D, 0x36, 0x95, \ - 0xA9, 0xE1, 0x36, 0x41, 0x14, 0x64, 0x33, 0xFB, \ - 0xCC, 0x93, 0x9D, 0xCE, 0x24, 0x9B, 0x3E, 0xF9, \ - 0x7D, 0x2F, 0xE3, 0x63, 0x63, 0x0C, 0x75, 0xD8, \ - 0xF6, 0x81, 0xB2, 0x02, 0xAE, 0xC4, 0x61, 0x7A, \ - 0xD3, 0xDF, 0x1E, 0xD5, 0xD5, 0xFD, 0x65, 0x61, \ - 0x24, 0x33, 0xF5, 0x1F, 0x5F, 0x06, 0x6E, 0xD0, \ - 0x85, 0x63, 0x65, 0x55, 0x3D, 0xED, 0x1A, 0xF3, \ - 0xB5, 0x57, 0x13, 0x5E, 0x7F, 0x57, 0xC9, 0x35, \ - 0x98, 0x4F, 0x0C, 0x70, 0xE0, 0xE6, 0x8B, 0x77, \ - 0xE2, 0xA6, 0x89, 0xDA, 0xF3, 0xEF, 0xE8, 0x72, \ - 0x1D, 0xF1, 0x58, 0xA1, 0x36, 0xAD, 0xE7, 0x35, \ - 0x30, 0xAC, 0xCA, 0x4F, 0x48, 0x3A, 0x79, 0x7A, \ - 0xBC, 0x0A, 0xB1, 0x82, 0xB3, 0x24, 0xFB, 0x61, \ - 0xD1, 0x08, 0xA9, 0x4B, 0xB2, 0xC8, 0xE3, 0xFB, \ - 0xB9, 0x6A, 0xDA, 0xB7, 0x60, 0xD7, 0xF4, 0x68, \ - 0x1D, 0x4F, 0x42, 0xA3, 0xDE, 0x39, 0x4D, 0xF4, \ - 0xAE, 0x56, 0xED, 0xE7, 0x63, 0x72, 0xBB, 0x19, \ - 0x0B, 0x07, 0xA7, 0xC8, 0xEE, 0x0A, 0x6D, 0x70, \ - 0x9E, 0x02, 0xFC, 0xE1, 0xCD, 0xF7, 0xE2, 0xEC, \ - 0xC0, 0x34, 0x04, 0xCD, 0x28, 0x34, 0x2F, 0x61, \ - 0x91, 0x72, 0xFE, 0x9C, 0xE9, 0x85, 0x83, 0xFF, \ - 0x8E, 0x4F, 0x12, 0x32, 0xEE, 0xF2, 0x81, 0x83, \ - 0xC3, 0xFE, 0x3B, 0x1B, 0x4C, 0x6F, 0xAD, 0x73, \ - 0x3B, 0xB5, 0xFC, 0xBC, 0x2E, 0xC2, 0x20, 0x05, \ - 0xC5, 0x8E, 0xF1, 0x83, 0x7D, 0x16, 0x83, 0xB2, \ - 0xC6, 0xF3, 0x4A, 0x26, 0xC1, 0xB2, 0xEF, 0xFA, \ - 0x88, 0x6B, 0x42, 0x38, 0x61, 0x1F, 0xCF, 0xDC, \ - 0xDE, 0x35, 0x5B, 0x3B, 0x65, 0x19, 0x03, 0x5B, \ - 0xBC, 0x34, 0xF4, 0xDE, 0xF9, 0x9C, 0x02, 0x38, \ - 0x61, 0xB4, 0x6F, 0xC9, 0xD6, 0xE6, 0xC9, 0x07, \ - 0x7A, 0xD9, 0x1D, 0x26, 0x91, 0xF7, 0xF7, 0xEE, \ - 0x59, 0x8C, 0xB0, 0xFA, 0xC1, 0x86, 0xD9, 0x1C, \ - 0xAE, 0xFE, 0x13, 0x09, 0x85, 0x13, 0x92, 0x70, \ - 0xB4, 0x13, 0x0C, 0x93, 0xBC, 0x43, 0x79, 0x44, \ - 0xF4, 0xFD, 0x44, 0x52, 0xE2, 0xD7, 0x4D, 0xD3, \ - 0x64, 0xF2, 0xE2, 0x1E, 0x71, 0xF5, 0x4B, 0xFF, \ - 0x5C, 0xAE, 0x82, 0xAB, 0x9C, 0x9D, 0xF6, 0x9E, \ - 0xE8, 0x6D, 0x2B, 0xC5, 0x22, 0x36, 0x3A, 0x0D, \ - 0xAB, 0xC5, 0x21, 0x97, 0x9B, 0x0D, 0xEA, 0xDA, \ - 0x1D, 0xBF, 0x9A, 0x42, 0xD5, 0xC4, 0x48, 0x4E, \ - 0x0A, 0xBC, 0xD0, 0x6B, 0xFA, 0x53, 0xDD, 0xEF, \ - 0x3C, 0x1B, 0x20, 0xEE, 0x3F, 0xD5, 0x9D, 0x7C, \ - 0x25, 0xE4, 0x1D, 0x2B, 0x66, 0x9E, 0x1E, 0xF1, \ - 0x6E, 0x6F, 0x52, 0xC3, 0x16, 0x4D, 0xF4, 0xFB, \ - 0x79, 0x30, 0xE9, 0xE4, 0xE5, 0x88, 0x57, 0xB6, \ - 0xAC, 0x7D, 0x5F, 0x42, 0xD6, 0x9F, 0x6D, 0x18, \ - 0x77, 0x63, 0xCF, 0x1D, 0x55, 0x03, 0x40, 0x04, \ - 0x87, 0xF5, 0x5B, 0xA5, 0x7E, 0x31, 0xCC, 0x7A, \ - 0x71, 0x35, 0xC8, 0x86, 0xEF, 0xB4, 0x31, 0x8A, \ - 0xED, 0x6A, 0x1E, 0x01, 0x2D, 0x9E, 0x68, 0x32, \ - 0xA9, 0x07, 0x60, 0x0A, 0x91, 0x81, 0x30, 0xC4, \ - 0x6D, 0xC7, 0x78, 0xF9, 0x71, 0xAD, 0x00, 0x38, \ - 0x09, 0x29, 0x99, 0xA3, 0x33, 0xCB, 0x8B, 0x7A, \ - 0x1A, 0x1D, 0xB9, 0x3D, 0x71, 0x40, 0x00, 0x3C, \ - 0x2A, 0x4E, 0xCE, 0xA9, 0xF9, 0x8D, 0x0A, 0xCC, \ - 0x0A, 0x82, 0x91, 0xCD, 0xCE, 0xC9, 0x7D, 0xCF, \ - 0x8E, 0xC9, 0xB5, 0x5A, 0x7F, 0x88, 0xA4, 0x6B, \ - 0x4D, 0xB5, 0xA8, 0x51, 0xF4, 0x41, 0x82, 0xE1, \ - 0xC6, 0x8A, 0x00, 0x7E, 0x5E, 0x0D, 0xD9, 0x02, \ - 0x0B, 0xFD, 0x64, 0xB6, 0x45, 0x03, 0x6C, 0x7A, \ - 0x4E, 0x67, 0x7D, 0x2C, 0x38, 0x53, 0x2A, 0x3A, \ - 0x23, 0xBA, 0x44, 0x42, 0xCA, 0xF5, 0x3E, 0xA6, \ - 0x3B, 0xB4, 0x54, 0x32, 0x9B, 0x76, 0x24, 0xC8, \ - 0x91, 0x7B, 0xDD, 0x64, 0xB1, 0xC0, 0xFD, 0x4C, \ - 0xB3, 0x8E, 0x8C, 0x33, 0x4C, 0x70, 0x1C, 0x3A, \ - 0xCD, 0xAD, 0x06, 0x57, 0xFC, 0xCF, 0xEC, 0x71, \ - 0x9B, 0x1F, 0x5C, 0x3E, 0x4E, 0x46, 0x04, 0x1F, \ - 0x38, 0x81, 0x47, 0xFB, 0x4C, 0xFD, 0xB4, 0x77, \ - 0xA5, 0x24, 0x71, 0xF7, 0xA9, 0xA9, 0x69, 0x10, \ - 0xB8, 0x55, 0x32, 0x2E, 0xDB, 0x63, 0x40, 0xD8, \ - 0xA0, 0x0E, 0xF0, 0x92, 0x35, 0x05, 0x11, 0xE3, \ - 0x0A, 0xBE, 0xC1, 0xFF, 0xF9, 0xE3, 0xA2, 0x6E, \ - 0x7F, 0xB2, 0x9F, 0x8C, 0x18, 0x30, 0x23, 0xC3, \ - 0x58, 0x7E, 0x38, 0xDA, 0x00, 0x77, 0xD9, 0xB4, \ - 0x76, 0x3E, 0x4E, 0x4B, 0x94, 0xB2, 0xBB, 0xC1, \ - 0x94, 0xC6, 0x65, 0x1E, 0x77, 0xCA, 0xF9, 0x92, \ - 0xEE, 0xAA, 0xC0, 0x23, 0x2A, 0x28, 0x1B, 0xF6, \ - 0xB3, 0xA7, 0x39, 0xC1, 0x22, 0x61, 0x16, 0x82, \ - 0x0A, 0xE8, 0xDB, 0x58, 0x47, 0xA6, 0x7C, 0xBE, \ - 0xF9, 0xC9, 0x09, 0x1B, 0x46, 0x2D, 0x53, 0x8C, \ - 0xD7, 0x2B, 0x03, 0x74, 0x6A, 0xE7, 0x7F, 0x5E, \ - 0x62, 0x29, 0x2C, 0x31, 0x15, 0x62, 0xA8, 0x46, \ - 0x50, 0x5D, 0xC8, 0x2D, 0xB8, 0x54, 0x33, 0x8A, \ - 0xE4, 0x9F, 0x52, 0x35, 0xC9, 0x5B, 0x91, 0x17, \ - 0x8C, 0xCF, 0x2D, 0xD5, 0xCA, 0xCE, 0xF4, 0x03, \ - 0xEC, 0x9D, 0x18, 0x10, 0xC6, 0x27, 0x2B, 0x04, \ - 0x5B, 0x3B, 0x71, 0xF9, 0xDC, 0x6B, 0x80, 0xD6, \ - 0x3F, 0xDD, 0x4A, 0x8E, 0x9A, 0xDB, 0x1E, 0x69, \ - 0x62, 0xA6, 0x95, 0x26, 0xD4, 0x31, 0x61, 0xC1, \ - 0xA4, 0x1D, 0x57, 0x0D, 0x79, 0x38, 0xDA, 0xD4, \ - 0xA4, 0x0E, 0x32, 0x9C, 0xD0, 0xE4, 0x0E, 0x65, \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF } - -#define MBEDTLS_DHM_RFC7919_FFDHE6144_G_BIN { 0x02 } - -#define MBEDTLS_DHM_RFC7919_FFDHE8192_P_BIN { \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, \ - 0xAD, 0xF8, 0x54, 0x58, 0xA2, 0xBB, 0x4A, 0x9A, \ - 0xAF, 0xDC, 0x56, 0x20, 0x27, 0x3D, 0x3C, 0xF1, \ - 0xD8, 0xB9, 0xC5, 0x83, 0xCE, 0x2D, 0x36, 0x95, \ - 0xA9, 0xE1, 0x36, 0x41, 0x14, 0x64, 0x33, 0xFB, \ - 0xCC, 0x93, 0x9D, 0xCE, 0x24, 0x9B, 0x3E, 0xF9, \ - 0x7D, 0x2F, 0xE3, 0x63, 0x63, 0x0C, 0x75, 0xD8, \ - 0xF6, 0x81, 0xB2, 0x02, 0xAE, 0xC4, 0x61, 0x7A, \ - 0xD3, 0xDF, 0x1E, 0xD5, 0xD5, 0xFD, 0x65, 0x61, \ - 0x24, 0x33, 0xF5, 0x1F, 0x5F, 0x06, 0x6E, 0xD0, \ - 0x85, 0x63, 0x65, 0x55, 0x3D, 0xED, 0x1A, 0xF3, \ - 0xB5, 0x57, 0x13, 0x5E, 0x7F, 0x57, 0xC9, 0x35, \ - 0x98, 0x4F, 0x0C, 0x70, 0xE0, 0xE6, 0x8B, 0x77, \ - 0xE2, 0xA6, 0x89, 0xDA, 0xF3, 0xEF, 0xE8, 0x72, \ - 0x1D, 0xF1, 0x58, 0xA1, 0x36, 0xAD, 0xE7, 0x35, \ - 0x30, 0xAC, 0xCA, 0x4F, 0x48, 0x3A, 0x79, 0x7A, \ - 0xBC, 0x0A, 0xB1, 0x82, 0xB3, 0x24, 0xFB, 0x61, \ - 0xD1, 0x08, 0xA9, 0x4B, 0xB2, 0xC8, 0xE3, 0xFB, \ - 0xB9, 0x6A, 0xDA, 0xB7, 0x60, 0xD7, 0xF4, 0x68, \ - 0x1D, 0x4F, 0x42, 0xA3, 0xDE, 0x39, 0x4D, 0xF4, \ - 0xAE, 0x56, 0xED, 0xE7, 0x63, 0x72, 0xBB, 0x19, \ - 0x0B, 0x07, 0xA7, 0xC8, 0xEE, 0x0A, 0x6D, 0x70, \ - 0x9E, 0x02, 0xFC, 0xE1, 0xCD, 0xF7, 0xE2, 0xEC, \ - 0xC0, 0x34, 0x04, 0xCD, 0x28, 0x34, 0x2F, 0x61, \ - 0x91, 0x72, 0xFE, 0x9C, 0xE9, 0x85, 0x83, 0xFF, \ - 0x8E, 0x4F, 0x12, 0x32, 0xEE, 0xF2, 0x81, 0x83, \ - 0xC3, 0xFE, 0x3B, 0x1B, 0x4C, 0x6F, 0xAD, 0x73, \ - 0x3B, 0xB5, 0xFC, 0xBC, 0x2E, 0xC2, 0x20, 0x05, \ - 0xC5, 0x8E, 0xF1, 0x83, 0x7D, 0x16, 0x83, 0xB2, \ - 0xC6, 0xF3, 0x4A, 0x26, 0xC1, 0xB2, 0xEF, 0xFA, \ - 0x88, 0x6B, 0x42, 0x38, 0x61, 0x1F, 0xCF, 0xDC, \ - 0xDE, 0x35, 0x5B, 0x3B, 0x65, 0x19, 0x03, 0x5B, \ - 0xBC, 0x34, 0xF4, 0xDE, 0xF9, 0x9C, 0x02, 0x38, \ - 0x61, 0xB4, 0x6F, 0xC9, 0xD6, 0xE6, 0xC9, 0x07, \ - 0x7A, 0xD9, 0x1D, 0x26, 0x91, 0xF7, 0xF7, 0xEE, \ - 0x59, 0x8C, 0xB0, 0xFA, 0xC1, 0x86, 0xD9, 0x1C, \ - 0xAE, 0xFE, 0x13, 0x09, 0x85, 0x13, 0x92, 0x70, \ - 0xB4, 0x13, 0x0C, 0x93, 0xBC, 0x43, 0x79, 0x44, \ - 0xF4, 0xFD, 0x44, 0x52, 0xE2, 0xD7, 0x4D, 0xD3, \ - 0x64, 0xF2, 0xE2, 0x1E, 0x71, 0xF5, 0x4B, 0xFF, \ - 0x5C, 0xAE, 0x82, 0xAB, 0x9C, 0x9D, 0xF6, 0x9E, \ - 0xE8, 0x6D, 0x2B, 0xC5, 0x22, 0x36, 0x3A, 0x0D, \ - 0xAB, 0xC5, 0x21, 0x97, 0x9B, 0x0D, 0xEA, 0xDA, \ - 0x1D, 0xBF, 0x9A, 0x42, 0xD5, 0xC4, 0x48, 0x4E, \ - 0x0A, 0xBC, 0xD0, 0x6B, 0xFA, 0x53, 0xDD, 0xEF, \ - 0x3C, 0x1B, 0x20, 0xEE, 0x3F, 0xD5, 0x9D, 0x7C, \ - 0x25, 0xE4, 0x1D, 0x2B, 0x66, 0x9E, 0x1E, 0xF1, \ - 0x6E, 0x6F, 0x52, 0xC3, 0x16, 0x4D, 0xF4, 0xFB, \ - 0x79, 0x30, 0xE9, 0xE4, 0xE5, 0x88, 0x57, 0xB6, \ - 0xAC, 0x7D, 0x5F, 0x42, 0xD6, 0x9F, 0x6D, 0x18, \ - 0x77, 0x63, 0xCF, 0x1D, 0x55, 0x03, 0x40, 0x04, \ - 0x87, 0xF5, 0x5B, 0xA5, 0x7E, 0x31, 0xCC, 0x7A, \ - 0x71, 0x35, 0xC8, 0x86, 0xEF, 0xB4, 0x31, 0x8A, \ - 0xED, 0x6A, 0x1E, 0x01, 0x2D, 0x9E, 0x68, 0x32, \ - 0xA9, 0x07, 0x60, 0x0A, 0x91, 0x81, 0x30, 0xC4, \ - 0x6D, 0xC7, 0x78, 0xF9, 0x71, 0xAD, 0x00, 0x38, \ - 0x09, 0x29, 0x99, 0xA3, 0x33, 0xCB, 0x8B, 0x7A, \ - 0x1A, 0x1D, 0xB9, 0x3D, 0x71, 0x40, 0x00, 0x3C, \ - 0x2A, 0x4E, 0xCE, 0xA9, 0xF9, 0x8D, 0x0A, 0xCC, \ - 0x0A, 0x82, 0x91, 0xCD, 0xCE, 0xC9, 0x7D, 0xCF, \ - 0x8E, 0xC9, 0xB5, 0x5A, 0x7F, 0x88, 0xA4, 0x6B, \ - 0x4D, 0xB5, 0xA8, 0x51, 0xF4, 0x41, 0x82, 0xE1, \ - 0xC6, 0x8A, 0x00, 0x7E, 0x5E, 0x0D, 0xD9, 0x02, \ - 0x0B, 0xFD, 0x64, 0xB6, 0x45, 0x03, 0x6C, 0x7A, \ - 0x4E, 0x67, 0x7D, 0x2C, 0x38, 0x53, 0x2A, 0x3A, \ - 0x23, 0xBA, 0x44, 0x42, 0xCA, 0xF5, 0x3E, 0xA6, \ - 0x3B, 0xB4, 0x54, 0x32, 0x9B, 0x76, 0x24, 0xC8, \ - 0x91, 0x7B, 0xDD, 0x64, 0xB1, 0xC0, 0xFD, 0x4C, \ - 0xB3, 0x8E, 0x8C, 0x33, 0x4C, 0x70, 0x1C, 0x3A, \ - 0xCD, 0xAD, 0x06, 0x57, 0xFC, 0xCF, 0xEC, 0x71, \ - 0x9B, 0x1F, 0x5C, 0x3E, 0x4E, 0x46, 0x04, 0x1F, \ - 0x38, 0x81, 0x47, 0xFB, 0x4C, 0xFD, 0xB4, 0x77, \ - 0xA5, 0x24, 0x71, 0xF7, 0xA9, 0xA9, 0x69, 0x10, \ - 0xB8, 0x55, 0x32, 0x2E, 0xDB, 0x63, 0x40, 0xD8, \ - 0xA0, 0x0E, 0xF0, 0x92, 0x35, 0x05, 0x11, 0xE3, \ - 0x0A, 0xBE, 0xC1, 0xFF, 0xF9, 0xE3, 0xA2, 0x6E, \ - 0x7F, 0xB2, 0x9F, 0x8C, 0x18, 0x30, 0x23, 0xC3, \ - 0x58, 0x7E, 0x38, 0xDA, 0x00, 0x77, 0xD9, 0xB4, \ - 0x76, 0x3E, 0x4E, 0x4B, 0x94, 0xB2, 0xBB, 0xC1, \ - 0x94, 0xC6, 0x65, 0x1E, 0x77, 0xCA, 0xF9, 0x92, \ - 0xEE, 0xAA, 0xC0, 0x23, 0x2A, 0x28, 0x1B, 0xF6, \ - 0xB3, 0xA7, 0x39, 0xC1, 0x22, 0x61, 0x16, 0x82, \ - 0x0A, 0xE8, 0xDB, 0x58, 0x47, 0xA6, 0x7C, 0xBE, \ - 0xF9, 0xC9, 0x09, 0x1B, 0x46, 0x2D, 0x53, 0x8C, \ - 0xD7, 0x2B, 0x03, 0x74, 0x6A, 0xE7, 0x7F, 0x5E, \ - 0x62, 0x29, 0x2C, 0x31, 0x15, 0x62, 0xA8, 0x46, \ - 0x50, 0x5D, 0xC8, 0x2D, 0xB8, 0x54, 0x33, 0x8A, \ - 0xE4, 0x9F, 0x52, 0x35, 0xC9, 0x5B, 0x91, 0x17, \ - 0x8C, 0xCF, 0x2D, 0xD5, 0xCA, 0xCE, 0xF4, 0x03, \ - 0xEC, 0x9D, 0x18, 0x10, 0xC6, 0x27, 0x2B, 0x04, \ - 0x5B, 0x3B, 0x71, 0xF9, 0xDC, 0x6B, 0x80, 0xD6, \ - 0x3F, 0xDD, 0x4A, 0x8E, 0x9A, 0xDB, 0x1E, 0x69, \ - 0x62, 0xA6, 0x95, 0x26, 0xD4, 0x31, 0x61, 0xC1, \ - 0xA4, 0x1D, 0x57, 0x0D, 0x79, 0x38, 0xDA, 0xD4, \ - 0xA4, 0x0E, 0x32, 0x9C, 0xCF, 0xF4, 0x6A, 0xAA, \ - 0x36, 0xAD, 0x00, 0x4C, 0xF6, 0x00, 0xC8, 0x38, \ - 0x1E, 0x42, 0x5A, 0x31, 0xD9, 0x51, 0xAE, 0x64, \ - 0xFD, 0xB2, 0x3F, 0xCE, 0xC9, 0x50, 0x9D, 0x43, \ - 0x68, 0x7F, 0xEB, 0x69, 0xED, 0xD1, 0xCC, 0x5E, \ - 0x0B, 0x8C, 0xC3, 0xBD, 0xF6, 0x4B, 0x10, 0xEF, \ - 0x86, 0xB6, 0x31, 0x42, 0xA3, 0xAB, 0x88, 0x29, \ - 0x55, 0x5B, 0x2F, 0x74, 0x7C, 0x93, 0x26, 0x65, \ - 0xCB, 0x2C, 0x0F, 0x1C, 0xC0, 0x1B, 0xD7, 0x02, \ - 0x29, 0x38, 0x88, 0x39, 0xD2, 0xAF, 0x05, 0xE4, \ - 0x54, 0x50, 0x4A, 0xC7, 0x8B, 0x75, 0x82, 0x82, \ - 0x28, 0x46, 0xC0, 0xBA, 0x35, 0xC3, 0x5F, 0x5C, \ - 0x59, 0x16, 0x0C, 0xC0, 0x46, 0xFD, 0x82, 0x51, \ - 0x54, 0x1F, 0xC6, 0x8C, 0x9C, 0x86, 0xB0, 0x22, \ - 0xBB, 0x70, 0x99, 0x87, 0x6A, 0x46, 0x0E, 0x74, \ - 0x51, 0xA8, 0xA9, 0x31, 0x09, 0x70, 0x3F, 0xEE, \ - 0x1C, 0x21, 0x7E, 0x6C, 0x38, 0x26, 0xE5, 0x2C, \ - 0x51, 0xAA, 0x69, 0x1E, 0x0E, 0x42, 0x3C, 0xFC, \ - 0x99, 0xE9, 0xE3, 0x16, 0x50, 0xC1, 0x21, 0x7B, \ - 0x62, 0x48, 0x16, 0xCD, 0xAD, 0x9A, 0x95, 0xF9, \ - 0xD5, 0xB8, 0x01, 0x94, 0x88, 0xD9, 0xC0, 0xA0, \ - 0xA1, 0xFE, 0x30, 0x75, 0xA5, 0x77, 0xE2, 0x31, \ - 0x83, 0xF8, 0x1D, 0x4A, 0x3F, 0x2F, 0xA4, 0x57, \ - 0x1E, 0xFC, 0x8C, 0xE0, 0xBA, 0x8A, 0x4F, 0xE8, \ - 0xB6, 0x85, 0x5D, 0xFE, 0x72, 0xB0, 0xA6, 0x6E, \ - 0xDE, 0xD2, 0xFB, 0xAB, 0xFB, 0xE5, 0x8A, 0x30, \ - 0xFA, 0xFA, 0xBE, 0x1C, 0x5D, 0x71, 0xA8, 0x7E, \ - 0x2F, 0x74, 0x1E, 0xF8, 0xC1, 0xFE, 0x86, 0xFE, \ - 0xA6, 0xBB, 0xFD, 0xE5, 0x30, 0x67, 0x7F, 0x0D, \ - 0x97, 0xD1, 0x1D, 0x49, 0xF7, 0xA8, 0x44, 0x3D, \ - 0x08, 0x22, 0xE5, 0x06, 0xA9, 0xF4, 0x61, 0x4E, \ - 0x01, 0x1E, 0x2A, 0x94, 0x83, 0x8F, 0xF8, 0x8C, \ - 0xD6, 0x8C, 0x8B, 0xB7, 0xC5, 0xC6, 0x42, 0x4C, \ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF } - -#define MBEDTLS_DHM_RFC7919_FFDHE8192_G_BIN { 0x02 } - -#endif /* dhm.h */ diff --git a/windows/mbedtls/ecdh.h b/windows/mbedtls/ecdh.h deleted file mode 100644 index 7fa7440a..00000000 --- a/windows/mbedtls/ecdh.h +++ /dev/null @@ -1,458 +0,0 @@ -/** - * \file ecdh.h - * - * \brief This file contains ECDH definitions and functions. - * - * The Elliptic Curve Diffie-Hellman (ECDH) protocol is an anonymous - * key agreement protocol allowing two parties to establish a shared - * secret over an insecure channel. Each party must have an - * elliptic-curve public–private key pair. - * - * For more information, see NIST SP 800-56A Rev. 2: Recommendation for - * Pair-Wise Key Establishment Schemes Using Discrete Logarithm - * Cryptography. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_ECDH_H -#define MBEDTLS_ECDH_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/ecp.h" - -/* - * Mbed TLS supports two formats for ECDH contexts (#mbedtls_ecdh_context - * defined in `ecdh.h`). For most applications, the choice of format makes - * no difference, since all library functions can work with either format, - * except that the new format is incompatible with MBEDTLS_ECP_RESTARTABLE. - - * The new format used when this option is disabled is smaller - * (56 bytes on a 32-bit platform). In future versions of the library, it - * will support alternative implementations of ECDH operations. - * The new format is incompatible with applications that access - * context fields directly and with restartable ECP operations. - */ - -#if defined(MBEDTLS_ECP_RESTARTABLE) -#define MBEDTLS_ECDH_LEGACY_CONTEXT -#else -#undef MBEDTLS_ECDH_LEGACY_CONTEXT -#endif - -#if defined(MBEDTLS_ECDH_VARIANT_EVEREST_ENABLED) -#undef MBEDTLS_ECDH_LEGACY_CONTEXT -#include "everest/everest.h" -#endif - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * Defines the source of the imported EC key. - */ -typedef enum -{ - MBEDTLS_ECDH_OURS, /**< Our key. */ - MBEDTLS_ECDH_THEIRS, /**< The key of the peer. */ -} mbedtls_ecdh_side; - -#if !defined(MBEDTLS_ECDH_LEGACY_CONTEXT) -/** - * Defines the ECDH implementation used. - * - * Later versions of the library may add new variants, therefore users should - * not make any assumptions about them. - */ -typedef enum -{ - MBEDTLS_ECDH_VARIANT_NONE = 0, /*!< Implementation not defined. */ - MBEDTLS_ECDH_VARIANT_MBEDTLS_2_0,/*!< The default Mbed TLS implementation */ -#if defined(MBEDTLS_ECDH_VARIANT_EVEREST_ENABLED) - MBEDTLS_ECDH_VARIANT_EVEREST /*!< Everest implementation */ -#endif -} mbedtls_ecdh_variant; - -/** - * The context used by the default ECDH implementation. - * - * Later versions might change the structure of this context, therefore users - * should not make any assumptions about the structure of - * mbedtls_ecdh_context_mbed. - */ -typedef struct mbedtls_ecdh_context_mbed -{ - mbedtls_ecp_group MBEDTLS_PRIVATE(grp); /*!< The elliptic curve used. */ - mbedtls_mpi MBEDTLS_PRIVATE(d); /*!< The private key. */ - mbedtls_ecp_point MBEDTLS_PRIVATE(Q); /*!< The public key. */ - mbedtls_ecp_point MBEDTLS_PRIVATE(Qp); /*!< The value of the public key of the peer. */ - mbedtls_mpi MBEDTLS_PRIVATE(z); /*!< The shared secret. */ -#if defined(MBEDTLS_ECP_RESTARTABLE) - mbedtls_ecp_restart_ctx MBEDTLS_PRIVATE(rs); /*!< The restart context for EC computations. */ -#endif -} mbedtls_ecdh_context_mbed; -#endif - -/** - * - * \warning Performing multiple operations concurrently on the same - * ECDSA context is not supported; objects of this type - * should not be shared between multiple threads. - * \brief The ECDH context structure. - */ -typedef struct mbedtls_ecdh_context -{ -#if defined(MBEDTLS_ECDH_LEGACY_CONTEXT) - mbedtls_ecp_group MBEDTLS_PRIVATE(grp); /*!< The elliptic curve used. */ - mbedtls_mpi MBEDTLS_PRIVATE(d); /*!< The private key. */ - mbedtls_ecp_point MBEDTLS_PRIVATE(Q); /*!< The public key. */ - mbedtls_ecp_point MBEDTLS_PRIVATE(Qp); /*!< The value of the public key of the peer. */ - mbedtls_mpi MBEDTLS_PRIVATE(z); /*!< The shared secret. */ - int MBEDTLS_PRIVATE(point_format); /*!< The format of point export in TLS messages. */ - mbedtls_ecp_point MBEDTLS_PRIVATE(Vi); /*!< The blinding value. */ - mbedtls_ecp_point MBEDTLS_PRIVATE(Vf); /*!< The unblinding value. */ - mbedtls_mpi MBEDTLS_PRIVATE(_d); /*!< The previous \p d. */ -#if defined(MBEDTLS_ECP_RESTARTABLE) - int MBEDTLS_PRIVATE(restart_enabled); /*!< The flag for restartable mode. */ - mbedtls_ecp_restart_ctx MBEDTLS_PRIVATE(rs); /*!< The restart context for EC computations. */ -#endif /* MBEDTLS_ECP_RESTARTABLE */ -#else - uint8_t MBEDTLS_PRIVATE(point_format); /*!< The format of point export in TLS messages - as defined in RFC 4492. */ - mbedtls_ecp_group_id MBEDTLS_PRIVATE(grp_id);/*!< The elliptic curve used. */ - mbedtls_ecdh_variant MBEDTLS_PRIVATE(var); /*!< The ECDH implementation/structure used. */ - union - { - mbedtls_ecdh_context_mbed MBEDTLS_PRIVATE(mbed_ecdh); -#if defined(MBEDTLS_ECDH_VARIANT_EVEREST_ENABLED) - mbedtls_ecdh_context_everest MBEDTLS_PRIVATE(everest_ecdh); -#endif - } MBEDTLS_PRIVATE(ctx); /*!< Implementation-specific context. The - context in use is specified by the \c var - field. */ -#if defined(MBEDTLS_ECP_RESTARTABLE) - uint8_t MBEDTLS_PRIVATE(restart_enabled); /*!< The flag for restartable mode. Functions of - an alternative implementation not supporting - restartable mode must return - MBEDTLS_ERR_PLATFORM_FEATURE_UNSUPPORTED error - if this flag is set. */ -#endif /* MBEDTLS_ECP_RESTARTABLE */ -#endif /* MBEDTLS_ECDH_LEGACY_CONTEXT */ -} -mbedtls_ecdh_context; - -/** - * \brief Check whether a given group can be used for ECDH. - * - * \param gid The ECP group ID to check. - * - * \return \c 1 if the group can be used, \c 0 otherwise - */ -int mbedtls_ecdh_can_do( mbedtls_ecp_group_id gid ); - -/** - * \brief This function generates an ECDH keypair on an elliptic - * curve. - * - * This function performs the first of two core computations - * implemented during the ECDH key exchange. The second core - * computation is performed by mbedtls_ecdh_compute_shared(). - * - * \see ecp.h - * - * \param grp The ECP group to use. This must be initialized and have - * domain parameters loaded, for example through - * mbedtls_ecp_load() or mbedtls_ecp_tls_read_group(). - * \param d The destination MPI (private key). - * This must be initialized. - * \param Q The destination point (public key). - * This must be initialized. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL in case \p f_rng doesn't need a context argument. - * - * \return \c 0 on success. - * \return Another \c MBEDTLS_ERR_ECP_XXX or - * \c MBEDTLS_MPI_XXX error code on failure. - */ -int mbedtls_ecdh_gen_public( mbedtls_ecp_group *grp, mbedtls_mpi *d, mbedtls_ecp_point *Q, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function computes the shared secret. - * - * This function performs the second of two core computations - * implemented during the ECDH key exchange. The first core - * computation is performed by mbedtls_ecdh_gen_public(). - * - * \see ecp.h - * - * \note If \p f_rng is not NULL, it is used to implement - * countermeasures against side-channel attacks. - * For more information, see mbedtls_ecp_mul(). - * - * \param grp The ECP group to use. This must be initialized and have - * domain parameters loaded, for example through - * mbedtls_ecp_load() or mbedtls_ecp_tls_read_group(). - * \param z The destination MPI (shared secret). - * This must be initialized. - * \param Q The public key from another party. - * This must be initialized. - * \param d Our secret exponent (private key). - * This must be initialized. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL if \p f_rng is \c NULL or doesn't need a - * context argument. - * - * \return \c 0 on success. - * \return Another \c MBEDTLS_ERR_ECP_XXX or - * \c MBEDTLS_MPI_XXX error code on failure. - */ -int mbedtls_ecdh_compute_shared( mbedtls_ecp_group *grp, mbedtls_mpi *z, - const mbedtls_ecp_point *Q, const mbedtls_mpi *d, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function initializes an ECDH context. - * - * \param ctx The ECDH context to initialize. This must not be \c NULL. - */ -void mbedtls_ecdh_init( mbedtls_ecdh_context *ctx ); - -/** - * \brief This function sets up the ECDH context with the information - * given. - * - * This function should be called after mbedtls_ecdh_init() but - * before mbedtls_ecdh_make_params(). There is no need to call - * this function before mbedtls_ecdh_read_params(). - * - * This is the first function used by a TLS server for ECDHE - * ciphersuites. - * - * \param ctx The ECDH context to set up. This must be initialized. - * \param grp_id The group id of the group to set up the context for. - * - * \return \c 0 on success. - */ -int mbedtls_ecdh_setup( mbedtls_ecdh_context *ctx, - mbedtls_ecp_group_id grp_id ); - -/** - * \brief This function frees a context. - * - * \param ctx The context to free. This may be \c NULL, in which - * case this function does nothing. If it is not \c NULL, - * it must point to an initialized ECDH context. - */ -void mbedtls_ecdh_free( mbedtls_ecdh_context *ctx ); - -/** - * \brief This function generates an EC key pair and exports its - * in the format used in a TLS ServerKeyExchange handshake - * message. - * - * This is the second function used by a TLS server for ECDHE - * ciphersuites. (It is called after mbedtls_ecdh_setup().) - * - * \see ecp.h - * - * \param ctx The ECDH context to use. This must be initialized - * and bound to a group, for example via mbedtls_ecdh_setup(). - * \param olen The address at which to store the number of Bytes written. - * \param buf The destination buffer. This must be a writable buffer of - * length \p blen Bytes. - * \param blen The length of the destination buffer \p buf in Bytes. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL in case \p f_rng doesn't need a context argument. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_IN_PROGRESS if maximum number of - * operations was reached: see \c mbedtls_ecp_set_max_ops(). - * \return Another \c MBEDTLS_ERR_ECP_XXX error code on failure. - */ -int mbedtls_ecdh_make_params( mbedtls_ecdh_context *ctx, size_t *olen, - unsigned char *buf, size_t blen, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function parses the ECDHE parameters in a - * TLS ServerKeyExchange handshake message. - * - * \note In a TLS handshake, this is the how the client - * sets up its ECDHE context from the server's public - * ECDHE key material. - * - * \see ecp.h - * - * \param ctx The ECDHE context to use. This must be initialized. - * \param buf On input, \c *buf must be the start of the input buffer. - * On output, \c *buf is updated to point to the end of the - * data that has been read. On success, this is the first byte - * past the end of the ServerKeyExchange parameters. - * On error, this is the point at which an error has been - * detected, which is usually not useful except to debug - * failures. - * \param end The end of the input buffer. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_ECP_XXX error code on failure. - * - */ -int mbedtls_ecdh_read_params( mbedtls_ecdh_context *ctx, - const unsigned char **buf, - const unsigned char *end ); - -/** - * \brief This function sets up an ECDH context from an EC key. - * - * It is used by clients and servers in place of the - * ServerKeyEchange for static ECDH, and imports ECDH - * parameters from the EC key information of a certificate. - * - * \see ecp.h - * - * \param ctx The ECDH context to set up. This must be initialized. - * \param key The EC key to use. This must be initialized. - * \param side Defines the source of the key. Possible values are: - * - #MBEDTLS_ECDH_OURS: The key is ours. - * - #MBEDTLS_ECDH_THEIRS: The key is that of the peer. - * - * \return \c 0 on success. - * \return Another \c MBEDTLS_ERR_ECP_XXX error code on failure. - * - */ -int mbedtls_ecdh_get_params( mbedtls_ecdh_context *ctx, - const mbedtls_ecp_keypair *key, - mbedtls_ecdh_side side ); - -/** - * \brief This function generates a public key and exports it - * as a TLS ClientKeyExchange payload. - * - * This is the second function used by a TLS client for ECDH(E) - * ciphersuites. - * - * \see ecp.h - * - * \param ctx The ECDH context to use. This must be initialized - * and bound to a group, the latter usually by - * mbedtls_ecdh_read_params(). - * \param olen The address at which to store the number of Bytes written. - * This must not be \c NULL. - * \param buf The destination buffer. This must be a writable buffer - * of length \p blen Bytes. - * \param blen The size of the destination buffer \p buf in Bytes. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL in case \p f_rng doesn't need a context argument. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_IN_PROGRESS if maximum number of - * operations was reached: see \c mbedtls_ecp_set_max_ops(). - * \return Another \c MBEDTLS_ERR_ECP_XXX error code on failure. - */ -int mbedtls_ecdh_make_public( mbedtls_ecdh_context *ctx, size_t *olen, - unsigned char *buf, size_t blen, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function parses and processes the ECDHE payload of a - * TLS ClientKeyExchange message. - * - * This is the third function used by a TLS server for ECDH(E) - * ciphersuites. (It is called after mbedtls_ecdh_setup() and - * mbedtls_ecdh_make_params().) - * - * \see ecp.h - * - * \param ctx The ECDH context to use. This must be initialized - * and bound to a group, for example via mbedtls_ecdh_setup(). - * \param buf The pointer to the ClientKeyExchange payload. This must - * be a readable buffer of length \p blen Bytes. - * \param blen The length of the input buffer \p buf in Bytes. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_ECP_XXX error code on failure. - */ -int mbedtls_ecdh_read_public( mbedtls_ecdh_context *ctx, - const unsigned char *buf, size_t blen ); - -/** - * \brief This function derives and exports the shared secret. - * - * This is the last function used by both TLS client - * and servers. - * - * \note If \p f_rng is not NULL, it is used to implement - * countermeasures against side-channel attacks. - * For more information, see mbedtls_ecp_mul(). - * - * \see ecp.h - - * \param ctx The ECDH context to use. This must be initialized - * and have its own private key generated and the peer's - * public key imported. - * \param olen The address at which to store the total number of - * Bytes written on success. This must not be \c NULL. - * \param buf The buffer to write the generated shared key to. This - * must be a writable buffer of size \p blen Bytes. - * \param blen The length of the destination buffer \p buf in Bytes. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG context. This may be \c NULL if \p f_rng - * doesn't need a context argument. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_IN_PROGRESS if maximum number of - * operations was reached: see \c mbedtls_ecp_set_max_ops(). - * \return Another \c MBEDTLS_ERR_ECP_XXX error code on failure. - */ -int mbedtls_ecdh_calc_secret( mbedtls_ecdh_context *ctx, size_t *olen, - unsigned char *buf, size_t blen, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -#if defined(MBEDTLS_ECP_RESTARTABLE) -/** - * \brief This function enables restartable EC computations for this - * context. (Default: disabled.) - * - * \see \c mbedtls_ecp_set_max_ops() - * - * \note It is not possible to safely disable restartable - * computations once enabled, except by free-ing the context, - * which cancels possible in-progress operations. - * - * \param ctx The ECDH context to use. This must be initialized. - */ -void mbedtls_ecdh_enable_restart( mbedtls_ecdh_context *ctx ); -#endif /* MBEDTLS_ECP_RESTARTABLE */ - -#ifdef __cplusplus -} -#endif - -#endif /* ecdh.h */ diff --git a/windows/mbedtls/ecdsa.h b/windows/mbedtls/ecdsa.h deleted file mode 100644 index 71b73eee..00000000 --- a/windows/mbedtls/ecdsa.h +++ /dev/null @@ -1,506 +0,0 @@ -/** - * \file ecdsa.h - * - * \brief This file contains ECDSA definitions and functions. - * - * The Elliptic Curve Digital Signature Algorithm (ECDSA) is defined in - * Standards for Efficient Cryptography Group (SECG): - * SEC1 Elliptic Curve Cryptography. - * The use of ECDSA for TLS is defined in RFC-4492: Elliptic Curve - * Cryptography (ECC) Cipher Suites for Transport Layer Security (TLS). - * - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_ECDSA_H -#define MBEDTLS_ECDSA_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/ecp.h" -#include "mbedtls/md.h" - -/** - * \brief Maximum ECDSA signature size for a given curve bit size - * - * \param bits Curve size in bits - * \return Maximum signature size in bytes - * - * \note This macro returns a compile-time constant if its argument - * is one. It may evaluate its argument multiple times. - */ -/* - * Ecdsa-Sig-Value ::= SEQUENCE { - * r INTEGER, - * s INTEGER - * } - * - * For each of r and s, the value (V) may include an extra initial "0" bit. - */ -#define MBEDTLS_ECDSA_MAX_SIG_LEN( bits ) \ - ( /*T,L of SEQUENCE*/ ( ( bits ) >= 61 * 8 ? 3 : 2 ) + \ - /*T,L of r,s*/ 2 * ( ( ( bits ) >= 127 * 8 ? 3 : 2 ) + \ - /*V of r,s*/ ( ( bits ) + 8 ) / 8 ) ) - -/** The maximal size of an ECDSA signature in Bytes. */ -#define MBEDTLS_ECDSA_MAX_LEN MBEDTLS_ECDSA_MAX_SIG_LEN( MBEDTLS_ECP_MAX_BITS ) - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief The ECDSA context structure. - * - * \warning Performing multiple operations concurrently on the same - * ECDSA context is not supported; objects of this type - * should not be shared between multiple threads. - */ -typedef mbedtls_ecp_keypair mbedtls_ecdsa_context; - -#if defined(MBEDTLS_ECP_RESTARTABLE) - -/** - * \brief Internal restart context for ecdsa_verify() - * - * \note Opaque struct, defined in ecdsa.c - */ -typedef struct mbedtls_ecdsa_restart_ver mbedtls_ecdsa_restart_ver_ctx; - -/** - * \brief Internal restart context for ecdsa_sign() - * - * \note Opaque struct, defined in ecdsa.c - */ -typedef struct mbedtls_ecdsa_restart_sig mbedtls_ecdsa_restart_sig_ctx; - -#if defined(MBEDTLS_ECDSA_DETERMINISTIC) -/** - * \brief Internal restart context for ecdsa_sign_det() - * - * \note Opaque struct, defined in ecdsa.c - */ -typedef struct mbedtls_ecdsa_restart_det mbedtls_ecdsa_restart_det_ctx; -#endif - -/** - * \brief General context for resuming ECDSA operations - */ -typedef struct -{ - mbedtls_ecp_restart_ctx MBEDTLS_PRIVATE(ecp); /*!< base context for ECP restart and - shared administrative info */ - mbedtls_ecdsa_restart_ver_ctx *MBEDTLS_PRIVATE(ver); /*!< ecdsa_verify() sub-context */ - mbedtls_ecdsa_restart_sig_ctx *MBEDTLS_PRIVATE(sig); /*!< ecdsa_sign() sub-context */ -#if defined(MBEDTLS_ECDSA_DETERMINISTIC) - mbedtls_ecdsa_restart_det_ctx *MBEDTLS_PRIVATE(det); /*!< ecdsa_sign_det() sub-context */ -#endif -} mbedtls_ecdsa_restart_ctx; - -#else /* MBEDTLS_ECP_RESTARTABLE */ - -/* Now we can declare functions that take a pointer to that */ -typedef void mbedtls_ecdsa_restart_ctx; - -#endif /* MBEDTLS_ECP_RESTARTABLE */ - -/** - * \brief This function checks whether a given group can be used - * for ECDSA. - * - * \param gid The ECP group ID to check. - * - * \return \c 1 if the group can be used, \c 0 otherwise - */ -int mbedtls_ecdsa_can_do( mbedtls_ecp_group_id gid ); - -/** - * \brief This function computes the ECDSA signature of a - * previously-hashed message. - * - * \note The deterministic version implemented in - * mbedtls_ecdsa_sign_det_ext() is usually preferred. - * - * \note If the bitlength of the message hash is larger than the - * bitlength of the group order, then the hash is truncated - * as defined in Standards for Efficient Cryptography Group - * (SECG): SEC1 Elliptic Curve Cryptography, section - * 4.1.3, step 5. - * - * \see ecp.h - * - * \param grp The context for the elliptic curve to use. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param r The MPI context in which to store the first part - * the signature. This must be initialized. - * \param s The MPI context in which to store the second part - * the signature. This must be initialized. - * \param d The private signing key. This must be initialized. - * \param buf The content to be signed. This is usually the hash of - * the original data to be signed. This must be a readable - * buffer of length \p blen Bytes. It may be \c NULL if - * \p blen is zero. - * \param blen The length of \p buf in Bytes. - * \param f_rng The RNG function. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL if \p f_rng doesn't need a context parameter. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_ECP_XXX - * or \c MBEDTLS_MPI_XXX error code on failure. - */ -int mbedtls_ecdsa_sign( mbedtls_ecp_group *grp, mbedtls_mpi *r, mbedtls_mpi *s, - const mbedtls_mpi *d, const unsigned char *buf, size_t blen, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng ); - -#if defined(MBEDTLS_ECDSA_DETERMINISTIC) -/** - * \brief This function computes the ECDSA signature of a - * previously-hashed message, deterministic version. - * - * For more information, see RFC-6979: Deterministic - * Usage of the Digital Signature Algorithm (DSA) and Elliptic - * Curve Digital Signature Algorithm (ECDSA). - * - * \note If the bitlength of the message hash is larger than the - * bitlength of the group order, then the hash is truncated as - * defined in Standards for Efficient Cryptography Group - * (SECG): SEC1 Elliptic Curve Cryptography, section - * 4.1.3, step 5. - * - * \see ecp.h - * - * \param grp The context for the elliptic curve to use. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param r The MPI context in which to store the first part - * the signature. This must be initialized. - * \param s The MPI context in which to store the second part - * the signature. This must be initialized. - * \param d The private signing key. This must be initialized - * and setup, for example through mbedtls_ecp_gen_privkey(). - * \param buf The hashed content to be signed. This must be a readable - * buffer of length \p blen Bytes. It may be \c NULL if - * \p blen is zero. - * \param blen The length of \p buf in Bytes. - * \param md_alg The hash algorithm used to hash the original data. - * \param f_rng_blind The RNG function used for blinding. This must not be - * \c NULL. - * \param p_rng_blind The RNG context to be passed to \p f_rng. This may be - * \c NULL if \p f_rng doesn't need a context parameter. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_ECP_XXX or \c MBEDTLS_MPI_XXX - * error code on failure. - */ -int mbedtls_ecdsa_sign_det_ext( mbedtls_ecp_group *grp, mbedtls_mpi *r, - mbedtls_mpi *s, const mbedtls_mpi *d, - const unsigned char *buf, size_t blen, - mbedtls_md_type_t md_alg, - int (*f_rng_blind)(void *, unsigned char *, size_t), - void *p_rng_blind ); -#endif /* MBEDTLS_ECDSA_DETERMINISTIC */ - -/** - * \brief This function verifies the ECDSA signature of a - * previously-hashed message. - * - * \note If the bitlength of the message hash is larger than the - * bitlength of the group order, then the hash is truncated as - * defined in Standards for Efficient Cryptography Group - * (SECG): SEC1 Elliptic Curve Cryptography, section - * 4.1.4, step 3. - * - * \see ecp.h - * - * \param grp The ECP group to use. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param buf The hashed content that was signed. This must be a readable - * buffer of length \p blen Bytes. It may be \c NULL if - * \p blen is zero. - * \param blen The length of \p buf in Bytes. - * \param Q The public key to use for verification. This must be - * initialized and setup. - * \param r The first integer of the signature. - * This must be initialized. - * \param s The second integer of the signature. - * This must be initialized. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA if the signature - * is invalid. - * \return An \c MBEDTLS_ERR_ECP_XXX or \c MBEDTLS_MPI_XXX - * error code on failure for any other reason. - */ -int mbedtls_ecdsa_verify( mbedtls_ecp_group *grp, - const unsigned char *buf, size_t blen, - const mbedtls_ecp_point *Q, const mbedtls_mpi *r, - const mbedtls_mpi *s); - -/** - * \brief This function computes the ECDSA signature and writes it - * to a buffer, serialized as defined in RFC-4492: - * Elliptic Curve Cryptography (ECC) Cipher Suites for - * Transport Layer Security (TLS). - * - * \warning It is not thread-safe to use the same context in - * multiple threads. - * - * \note The deterministic version is used if - * #MBEDTLS_ECDSA_DETERMINISTIC is defined. For more - * information, see RFC-6979: Deterministic Usage - * of the Digital Signature Algorithm (DSA) and Elliptic - * Curve Digital Signature Algorithm (ECDSA). - * - * \note If the bitlength of the message hash is larger than the - * bitlength of the group order, then the hash is truncated as - * defined in Standards for Efficient Cryptography Group - * (SECG): SEC1 Elliptic Curve Cryptography, section - * 4.1.3, step 5. - * - * \see ecp.h - * - * \param ctx The ECDSA context to use. This must be initialized - * and have a group and private key bound to it, for example - * via mbedtls_ecdsa_genkey() or mbedtls_ecdsa_from_keypair(). - * \param md_alg The message digest that was used to hash the message. - * \param hash The message hash to be signed. This must be a readable - * buffer of length \p blen Bytes. - * \param hlen The length of the hash \p hash in Bytes. - * \param sig The buffer to which to write the signature. This must be a - * writable buffer of length at least twice as large as the - * size of the curve used, plus 9. For example, 73 Bytes if - * a 256-bit curve is used. A buffer length of - * #MBEDTLS_ECDSA_MAX_LEN is always safe. - * \param sig_size The size of the \p sig buffer in bytes. - * \param slen The address at which to store the actual length of - * the signature written. Must not be \c NULL. - * \param f_rng The RNG function. This must not be \c NULL if - * #MBEDTLS_ECDSA_DETERMINISTIC is unset. Otherwise, - * it is used only for blinding and may be set to \c NULL, but - * doing so is DEPRECATED. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL if \p f_rng is \c NULL or doesn't use a context. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_ECP_XXX, \c MBEDTLS_ERR_MPI_XXX or - * \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_ecdsa_write_signature( mbedtls_ecdsa_context *ctx, - mbedtls_md_type_t md_alg, - const unsigned char *hash, size_t hlen, - unsigned char *sig, size_t sig_size, size_t *slen, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function computes the ECDSA signature and writes it - * to a buffer, in a restartable way. - * - * \see \c mbedtls_ecdsa_write_signature() - * - * \note This function is like \c mbedtls_ecdsa_write_signature() - * but it can return early and restart according to the limit - * set with \c mbedtls_ecp_set_max_ops() to reduce blocking. - * - * \param ctx The ECDSA context to use. This must be initialized - * and have a group and private key bound to it, for example - * via mbedtls_ecdsa_genkey() or mbedtls_ecdsa_from_keypair(). - * \param md_alg The message digest that was used to hash the message. - * \param hash The message hash to be signed. This must be a readable - * buffer of length \p blen Bytes. - * \param hlen The length of the hash \p hash in Bytes. - * \param sig The buffer to which to write the signature. This must be a - * writable buffer of length at least twice as large as the - * size of the curve used, plus 9. For example, 73 Bytes if - * a 256-bit curve is used. A buffer length of - * #MBEDTLS_ECDSA_MAX_LEN is always safe. - * \param sig_size The size of the \p sig buffer in bytes. - * \param slen The address at which to store the actual length of - * the signature written. Must not be \c NULL. - * \param f_rng The RNG function. This must not be \c NULL if - * #MBEDTLS_ECDSA_DETERMINISTIC is unset. Otherwise, - * it is unused and may be set to \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL if \p f_rng is \c NULL or doesn't use a context. - * \param rs_ctx The restart context to use. This may be \c NULL to disable - * restarting. If it is not \c NULL, it must point to an - * initialized restart context. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_IN_PROGRESS if maximum number of - * operations was reached: see \c mbedtls_ecp_set_max_ops(). - * \return Another \c MBEDTLS_ERR_ECP_XXX, \c MBEDTLS_ERR_MPI_XXX or - * \c MBEDTLS_ERR_ASN1_XXX error code on failure. - */ -int mbedtls_ecdsa_write_signature_restartable( mbedtls_ecdsa_context *ctx, - mbedtls_md_type_t md_alg, - const unsigned char *hash, size_t hlen, - unsigned char *sig, size_t sig_size, size_t *slen, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - mbedtls_ecdsa_restart_ctx *rs_ctx ); - -/** - * \brief This function reads and verifies an ECDSA signature. - * - * \note If the bitlength of the message hash is larger than the - * bitlength of the group order, then the hash is truncated as - * defined in Standards for Efficient Cryptography Group - * (SECG): SEC1 Elliptic Curve Cryptography, section - * 4.1.4, step 3. - * - * \see ecp.h - * - * \param ctx The ECDSA context to use. This must be initialized - * and have a group and public key bound to it. - * \param hash The message hash that was signed. This must be a readable - * buffer of length \p size Bytes. - * \param hlen The size of the hash \p hash. - * \param sig The signature to read and verify. This must be a readable - * buffer of length \p slen Bytes. - * \param slen The size of \p sig in Bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA if signature is invalid. - * \return #MBEDTLS_ERR_ECP_SIG_LEN_MISMATCH if there is a valid - * signature in \p sig, but its length is less than \p siglen. - * \return An \c MBEDTLS_ERR_ECP_XXX or \c MBEDTLS_ERR_MPI_XXX - * error code on failure for any other reason. - */ -int mbedtls_ecdsa_read_signature( mbedtls_ecdsa_context *ctx, - const unsigned char *hash, size_t hlen, - const unsigned char *sig, size_t slen ); - -/** - * \brief This function reads and verifies an ECDSA signature, - * in a restartable way. - * - * \see \c mbedtls_ecdsa_read_signature() - * - * \note This function is like \c mbedtls_ecdsa_read_signature() - * but it can return early and restart according to the limit - * set with \c mbedtls_ecp_set_max_ops() to reduce blocking. - * - * \param ctx The ECDSA context to use. This must be initialized - * and have a group and public key bound to it. - * \param hash The message hash that was signed. This must be a readable - * buffer of length \p size Bytes. - * \param hlen The size of the hash \p hash. - * \param sig The signature to read and verify. This must be a readable - * buffer of length \p slen Bytes. - * \param slen The size of \p sig in Bytes. - * \param rs_ctx The restart context to use. This may be \c NULL to disable - * restarting. If it is not \c NULL, it must point to an - * initialized restart context. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA if signature is invalid. - * \return #MBEDTLS_ERR_ECP_SIG_LEN_MISMATCH if there is a valid - * signature in \p sig, but its length is less than \p siglen. - * \return #MBEDTLS_ERR_ECP_IN_PROGRESS if maximum number of - * operations was reached: see \c mbedtls_ecp_set_max_ops(). - * \return Another \c MBEDTLS_ERR_ECP_XXX or \c MBEDTLS_ERR_MPI_XXX - * error code on failure for any other reason. - */ -int mbedtls_ecdsa_read_signature_restartable( mbedtls_ecdsa_context *ctx, - const unsigned char *hash, size_t hlen, - const unsigned char *sig, size_t slen, - mbedtls_ecdsa_restart_ctx *rs_ctx ); - -/** - * \brief This function generates an ECDSA keypair on the given curve. - * - * \see ecp.h - * - * \param ctx The ECDSA context to store the keypair in. - * This must be initialized. - * \param gid The elliptic curve to use. One of the various - * \c MBEDTLS_ECP_DP_XXX macros depending on configuration. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL if \p f_rng doesn't need a context argument. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_ECP_XXX code on failure. - */ -int mbedtls_ecdsa_genkey( mbedtls_ecdsa_context *ctx, mbedtls_ecp_group_id gid, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng ); - -/** - * \brief This function sets up an ECDSA context from an EC key pair. - * - * \see ecp.h - * - * \param ctx The ECDSA context to setup. This must be initialized. - * \param key The EC key to use. This must be initialized and hold - * a private-public key pair or a public key. In the former - * case, the ECDSA context may be used for signature creation - * and verification after this call. In the latter case, it - * may be used for signature verification. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_ECP_XXX code on failure. - */ -int mbedtls_ecdsa_from_keypair( mbedtls_ecdsa_context *ctx, - const mbedtls_ecp_keypair *key ); - -/** - * \brief This function initializes an ECDSA context. - * - * \param ctx The ECDSA context to initialize. - * This must not be \c NULL. - */ -void mbedtls_ecdsa_init( mbedtls_ecdsa_context *ctx ); - -/** - * \brief This function frees an ECDSA context. - * - * \param ctx The ECDSA context to free. This may be \c NULL, - * in which case this function does nothing. If it - * is not \c NULL, it must be initialized. - */ -void mbedtls_ecdsa_free( mbedtls_ecdsa_context *ctx ); - -#if defined(MBEDTLS_ECP_RESTARTABLE) -/** - * \brief Initialize a restart context. - * - * \param ctx The restart context to initialize. - * This must not be \c NULL. - */ -void mbedtls_ecdsa_restart_init( mbedtls_ecdsa_restart_ctx *ctx ); - -/** - * \brief Free the components of a restart context. - * - * \param ctx The restart context to free. This may be \c NULL, - * in which case this function does nothing. If it - * is not \c NULL, it must be initialized. - */ -void mbedtls_ecdsa_restart_free( mbedtls_ecdsa_restart_ctx *ctx ); -#endif /* MBEDTLS_ECP_RESTARTABLE */ - -#ifdef __cplusplus -} -#endif - -#endif /* ecdsa.h */ diff --git a/windows/mbedtls/ecjpake.h b/windows/mbedtls/ecjpake.h deleted file mode 100644 index a73f6249..00000000 --- a/windows/mbedtls/ecjpake.h +++ /dev/null @@ -1,287 +0,0 @@ -/** - * \file ecjpake.h - * - * \brief Elliptic curve J-PAKE - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_ECJPAKE_H -#define MBEDTLS_ECJPAKE_H -#include "mbedtls/private_access.h" - -/* - * J-PAKE is a password-authenticated key exchange that allows deriving a - * strong shared secret from a (potentially low entropy) pre-shared - * passphrase, with forward secrecy and mutual authentication. - * https://en.wikipedia.org/wiki/Password_Authenticated_Key_Exchange_by_Juggling - * - * This file implements the Elliptic Curve variant of J-PAKE, - * as defined in Chapter 7.4 of the Thread v1.0 Specification, - * available to members of the Thread Group http://threadgroup.org/ - * - * As the J-PAKE algorithm is inherently symmetric, so is our API. - * Each party needs to send its first round message, in any order, to the - * other party, then each sends its second round message, in any order. - * The payloads are serialized in a way suitable for use in TLS, but could - * also be use outside TLS. - */ -#include "mbedtls/build_info.h" - -#include "mbedtls/ecp.h" -#include "mbedtls/md.h" - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * Roles in the EC J-PAKE exchange - */ -typedef enum { - MBEDTLS_ECJPAKE_CLIENT = 0, /**< Client */ - MBEDTLS_ECJPAKE_SERVER, /**< Server */ -} mbedtls_ecjpake_role; - -#if !defined(MBEDTLS_ECJPAKE_ALT) -/** - * EC J-PAKE context structure. - * - * J-PAKE is a symmetric protocol, except for the identifiers used in - * Zero-Knowledge Proofs, and the serialization of the second message - * (KeyExchange) as defined by the Thread spec. - * - * In order to benefit from this symmetry, we choose a different naming - * convetion from the Thread v1.0 spec. Correspondance is indicated in the - * description as a pair C: client name, S: server name - */ -typedef struct mbedtls_ecjpake_context -{ - const mbedtls_md_info_t *MBEDTLS_PRIVATE(md_info); /**< Hash to use */ - mbedtls_ecp_group MBEDTLS_PRIVATE(grp); /**< Elliptic curve */ - mbedtls_ecjpake_role MBEDTLS_PRIVATE(role); /**< Are we client or server? */ - int MBEDTLS_PRIVATE(point_format); /**< Format for point export */ - - mbedtls_ecp_point MBEDTLS_PRIVATE(Xm1); /**< My public key 1 C: X1, S: X3 */ - mbedtls_ecp_point MBEDTLS_PRIVATE(Xm2); /**< My public key 2 C: X2, S: X4 */ - mbedtls_ecp_point MBEDTLS_PRIVATE(Xp1); /**< Peer public key 1 C: X3, S: X1 */ - mbedtls_ecp_point MBEDTLS_PRIVATE(Xp2); /**< Peer public key 2 C: X4, S: X2 */ - mbedtls_ecp_point MBEDTLS_PRIVATE(Xp); /**< Peer public key C: Xs, S: Xc */ - - mbedtls_mpi MBEDTLS_PRIVATE(xm1); /**< My private key 1 C: x1, S: x3 */ - mbedtls_mpi MBEDTLS_PRIVATE(xm2); /**< My private key 2 C: x2, S: x4 */ - - mbedtls_mpi MBEDTLS_PRIVATE(s); /**< Pre-shared secret (passphrase) */ -} mbedtls_ecjpake_context; - -#else /* MBEDTLS_ECJPAKE_ALT */ -#include "ecjpake_alt.h" -#endif /* MBEDTLS_ECJPAKE_ALT */ - -/** - * \brief Initialize an ECJPAKE context. - * - * \param ctx The ECJPAKE context to initialize. - * This must not be \c NULL. - */ -void mbedtls_ecjpake_init( mbedtls_ecjpake_context *ctx ); - -/** - * \brief Set up an ECJPAKE context for use. - * - * \note Currently the only values for hash/curve allowed by the - * standard are #MBEDTLS_MD_SHA256/#MBEDTLS_ECP_DP_SECP256R1. - * - * \param ctx The ECJPAKE context to set up. This must be initialized. - * \param role The role of the caller. This must be either - * #MBEDTLS_ECJPAKE_CLIENT or #MBEDTLS_ECJPAKE_SERVER. - * \param hash The identifier of the hash function to use, - * for example #MBEDTLS_MD_SHA256. - * \param curve The identifier of the elliptic curve to use, - * for example #MBEDTLS_ECP_DP_SECP256R1. - * \param secret The pre-shared secret (passphrase). This must be - * a readable buffer of length \p len Bytes. It need - * only be valid for the duration of this call. - * \param len The length of the pre-shared secret \p secret. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_ecjpake_setup( mbedtls_ecjpake_context *ctx, - mbedtls_ecjpake_role role, - mbedtls_md_type_t hash, - mbedtls_ecp_group_id curve, - const unsigned char *secret, - size_t len ); - -/** - * \brief Set the point format for future reads and writes. - * - * \param ctx The ECJPAKE context to configure. - * \param point_format The point format to use: - * #MBEDTLS_ECP_PF_UNCOMPRESSED (default) - * or #MBEDTLS_ECP_PF_COMPRESSED. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA if \p point_format - * is invalid. - */ -int mbedtls_ecjpake_set_point_format( mbedtls_ecjpake_context *ctx, - int point_format ); - -/** - * \brief Check if an ECJPAKE context is ready for use. - * - * \param ctx The ECJPAKE context to check. This must be - * initialized. - * - * \return \c 0 if the context is ready for use. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA otherwise. - */ -int mbedtls_ecjpake_check( const mbedtls_ecjpake_context *ctx ); - -/** - * \brief Generate and write the first round message - * (TLS: contents of the Client/ServerHello extension, - * excluding extension type and length bytes). - * - * \param ctx The ECJPAKE context to use. This must be - * initialized and set up. - * \param buf The buffer to write the contents to. This must be a - * writable buffer of length \p len Bytes. - * \param len The length of \p buf in Bytes. - * \param olen The address at which to store the total number - * of Bytes written to \p buf. This must not be \c NULL. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG parameter to be passed to \p f_rng. This - * may be \c NULL if \p f_rng doesn't use a context. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_ecjpake_write_round_one( mbedtls_ecjpake_context *ctx, - unsigned char *buf, size_t len, size_t *olen, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief Read and process the first round message - * (TLS: contents of the Client/ServerHello extension, - * excluding extension type and length bytes). - * - * \param ctx The ECJPAKE context to use. This must be initialized - * and set up. - * \param buf The buffer holding the first round message. This must - * be a readable buffer of length \p len Bytes. - * \param len The length in Bytes of \p buf. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_ecjpake_read_round_one( mbedtls_ecjpake_context *ctx, - const unsigned char *buf, - size_t len ); - -/** - * \brief Generate and write the second round message - * (TLS: contents of the Client/ServerKeyExchange). - * - * \param ctx The ECJPAKE context to use. This must be initialized, - * set up, and already have performed round one. - * \param buf The buffer to write the round two contents to. - * This must be a writable buffer of length \p len Bytes. - * \param len The size of \p buf in Bytes. - * \param olen The address at which to store the total number of Bytes - * written to \p buf. This must not be \c NULL. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG parameter to be passed to \p f_rng. This - * may be \c NULL if \p f_rng doesn't use a context. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_ecjpake_write_round_two( mbedtls_ecjpake_context *ctx, - unsigned char *buf, size_t len, size_t *olen, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief Read and process the second round message - * (TLS: contents of the Client/ServerKeyExchange). - * - * \param ctx The ECJPAKE context to use. This must be initialized - * and set up and already have performed round one. - * \param buf The buffer holding the second round message. This must - * be a readable buffer of length \p len Bytes. - * \param len The length in Bytes of \p buf. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_ecjpake_read_round_two( mbedtls_ecjpake_context *ctx, - const unsigned char *buf, - size_t len ); - -/** - * \brief Derive the shared secret - * (TLS: Pre-Master Secret). - * - * \param ctx The ECJPAKE context to use. This must be initialized, - * set up and have performed both round one and two. - * \param buf The buffer to write the derived secret to. This must - * be a writable buffer of length \p len Bytes. - * \param len The length of \p buf in Bytes. - * \param olen The address at which to store the total number of Bytes - * written to \p buf. This must not be \c NULL. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG parameter to be passed to \p f_rng. This - * may be \c NULL if \p f_rng doesn't use a context. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_ecjpake_derive_secret( mbedtls_ecjpake_context *ctx, - unsigned char *buf, size_t len, size_t *olen, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This clears an ECJPAKE context and frees any - * embedded data structure. - * - * \param ctx The ECJPAKE context to free. This may be \c NULL, - * in which case this function does nothing. If it is not - * \c NULL, it must point to an initialized ECJPAKE context. - */ -void mbedtls_ecjpake_free( mbedtls_ecjpake_context *ctx ); - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief Checkup routine - * - * \return 0 if successful, or 1 if a test failed - */ -int mbedtls_ecjpake_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - - -#endif /* ecjpake.h */ diff --git a/windows/mbedtls/ecp.h b/windows/mbedtls/ecp.h deleted file mode 100644 index b87114bc..00000000 --- a/windows/mbedtls/ecp.h +++ /dev/null @@ -1,1286 +0,0 @@ -/** - * \file ecp.h - * - * \brief This file provides an API for Elliptic Curves over GF(P) (ECP). - * - * The use of ECP in cryptography and TLS is defined in - * Standards for Efficient Cryptography Group (SECG): SEC1 - * Elliptic Curve Cryptography and - * RFC-4492: Elliptic Curve Cryptography (ECC) Cipher Suites - * for Transport Layer Security (TLS). - * - * RFC-2409: The Internet Key Exchange (IKE) defines ECP - * group types. - * - */ - -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_ECP_H -#define MBEDTLS_ECP_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/bignum.h" - -/* - * ECP error codes - */ -#define MBEDTLS_ERR_ECP_BAD_INPUT_DATA -0x4F80 /**< Bad input parameters to function. */ -#define MBEDTLS_ERR_ECP_BUFFER_TOO_SMALL -0x4F00 /**< The buffer is too small to write to. */ -#define MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE -0x4E80 /**< The requested feature is not available, for example, the requested curve is not supported. */ -#define MBEDTLS_ERR_ECP_VERIFY_FAILED -0x4E00 /**< The signature is not valid. */ -#define MBEDTLS_ERR_ECP_ALLOC_FAILED -0x4D80 /**< Memory allocation failed. */ -#define MBEDTLS_ERR_ECP_RANDOM_FAILED -0x4D00 /**< Generation of random value, such as ephemeral key, failed. */ -#define MBEDTLS_ERR_ECP_INVALID_KEY -0x4C80 /**< Invalid private or public key. */ -#define MBEDTLS_ERR_ECP_SIG_LEN_MISMATCH -0x4C00 /**< The buffer contains a valid signature followed by more data. */ -#define MBEDTLS_ERR_ECP_IN_PROGRESS -0x4B00 /**< Operation in progress, call again with the same parameters to continue. */ - -/* Flags indicating whether to include code that is specific to certain - * types of curves. These flags are for internal library use only. */ -#if defined(MBEDTLS_ECP_DP_SECP192R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP224R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP256R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP384R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP521R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_BP256R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_BP384R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_BP512R1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP192K1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP224K1_ENABLED) || \ - defined(MBEDTLS_ECP_DP_SECP256K1_ENABLED) -#define MBEDTLS_ECP_SHORT_WEIERSTRASS_ENABLED -#endif -#if defined(MBEDTLS_ECP_DP_CURVE25519_ENABLED) || \ - defined(MBEDTLS_ECP_DP_CURVE448_ENABLED) -#define MBEDTLS_ECP_MONTGOMERY_ENABLED -#endif - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * Domain-parameter identifiers: curve, subgroup, and generator. - * - * \note Only curves over prime fields are supported. - * - * \warning This library does not support validation of arbitrary domain - * parameters. Therefore, only standardized domain parameters from trusted - * sources should be used. See mbedtls_ecp_group_load(). - */ -/* Note: when adding a new curve: - * - Add it at the end of this enum, otherwise you'll break the ABI by - * changing the numerical value for existing curves. - * - Increment MBEDTLS_ECP_DP_MAX below if needed. - * - Update the calculation of MBEDTLS_ECP_MAX_BITS below. - * - Add the corresponding MBEDTLS_ECP_DP_xxx_ENABLED macro definition to - * mbedtls_config.h. - * - List the curve as a dependency of MBEDTLS_ECP_C and - * MBEDTLS_ECDSA_C if supported in check_config.h. - * - Add the curve to the appropriate curve type macro - * MBEDTLS_ECP_yyy_ENABLED above. - * - Add the necessary definitions to ecp_curves.c. - * - Add the curve to the ecp_supported_curves array in ecp.c. - * - Add the curve to applicable profiles in x509_crt.c. - * - Add the curve to applicable presets in ssl_tls.c. - */ -typedef enum -{ - MBEDTLS_ECP_DP_NONE = 0, /*!< Curve not defined. */ - MBEDTLS_ECP_DP_SECP192R1, /*!< Domain parameters for the 192-bit curve defined by FIPS 186-4 and SEC1. */ - MBEDTLS_ECP_DP_SECP224R1, /*!< Domain parameters for the 224-bit curve defined by FIPS 186-4 and SEC1. */ - MBEDTLS_ECP_DP_SECP256R1, /*!< Domain parameters for the 256-bit curve defined by FIPS 186-4 and SEC1. */ - MBEDTLS_ECP_DP_SECP384R1, /*!< Domain parameters for the 384-bit curve defined by FIPS 186-4 and SEC1. */ - MBEDTLS_ECP_DP_SECP521R1, /*!< Domain parameters for the 521-bit curve defined by FIPS 186-4 and SEC1. */ - MBEDTLS_ECP_DP_BP256R1, /*!< Domain parameters for 256-bit Brainpool curve. */ - MBEDTLS_ECP_DP_BP384R1, /*!< Domain parameters for 384-bit Brainpool curve. */ - MBEDTLS_ECP_DP_BP512R1, /*!< Domain parameters for 512-bit Brainpool curve. */ - MBEDTLS_ECP_DP_CURVE25519, /*!< Domain parameters for Curve25519. */ - MBEDTLS_ECP_DP_SECP192K1, /*!< Domain parameters for 192-bit "Koblitz" curve. */ - MBEDTLS_ECP_DP_SECP224K1, /*!< Domain parameters for 224-bit "Koblitz" curve. */ - MBEDTLS_ECP_DP_SECP256K1, /*!< Domain parameters for 256-bit "Koblitz" curve. */ - MBEDTLS_ECP_DP_CURVE448, /*!< Domain parameters for Curve448. */ -} mbedtls_ecp_group_id; - -/** - * The number of supported curves, plus one for #MBEDTLS_ECP_DP_NONE. - * - * \note Montgomery curves are currently excluded. - */ -#define MBEDTLS_ECP_DP_MAX 12 - -/* - * Curve types - */ -typedef enum -{ - MBEDTLS_ECP_TYPE_NONE = 0, - MBEDTLS_ECP_TYPE_SHORT_WEIERSTRASS, /* y^2 = x^3 + a x + b */ - MBEDTLS_ECP_TYPE_MONTGOMERY, /* y^2 = x^3 + a x^2 + x */ -} mbedtls_ecp_curve_type; - -/** - * Curve information, for use by other modules. - */ -typedef struct mbedtls_ecp_curve_info -{ - mbedtls_ecp_group_id MBEDTLS_PRIVATE(grp_id); /*!< An internal identifier. */ - uint16_t MBEDTLS_PRIVATE(tls_id); /*!< The TLS NamedCurve identifier. */ - uint16_t MBEDTLS_PRIVATE(bit_size); /*!< The curve size in bits. */ - const char *MBEDTLS_PRIVATE(name); /*!< A human-friendly name. */ -} mbedtls_ecp_curve_info; - -/** - * \brief The ECP point structure, in Jacobian coordinates. - * - * \note All functions expect and return points satisfying - * the following condition: Z == 0 or - * Z == 1. Other values of \p Z are - * used only by internal functions. - * The point is zero, or "at infinity", if Z == 0. - * Otherwise, \p X and \p Y are its standard (affine) - * coordinates. - */ -typedef struct mbedtls_ecp_point -{ - mbedtls_mpi MBEDTLS_PRIVATE(X); /*!< The X coordinate of the ECP point. */ - mbedtls_mpi MBEDTLS_PRIVATE(Y); /*!< The Y coordinate of the ECP point. */ - mbedtls_mpi MBEDTLS_PRIVATE(Z); /*!< The Z coordinate of the ECP point. */ -} -mbedtls_ecp_point; - -#if !defined(MBEDTLS_ECP_ALT) -/* - * default mbed TLS elliptic curve arithmetic implementation - * - * (in case MBEDTLS_ECP_ALT is defined then the developer has to provide an - * alternative implementation for the whole module and it will replace this - * one.) - */ - -/** - * \brief The ECP group structure. - * - * We consider two types of curve equations: - *
  • Short Weierstrass: y^2 = x^3 + A x + B mod P - * (SEC1 + RFC-4492)
  • - *
  • Montgomery: y^2 = x^3 + A x^2 + x mod P (Curve25519, - * Curve448)
- * In both cases, the generator (\p G) for a prime-order subgroup is fixed. - * - * For Short Weierstrass, this subgroup is the whole curve, and its - * cardinality is denoted by \p N. Our code requires that \p N is an - * odd prime as mbedtls_ecp_mul() requires an odd number, and - * mbedtls_ecdsa_sign() requires that it is prime for blinding purposes. - * - * For Montgomery curves, we do not store \p A, but (A + 2) / 4, - * which is the quantity used in the formulas. Additionally, \p nbits is - * not the size of \p N but the required size for private keys. - * - * If \p modp is NULL, reduction modulo \p P is done using a generic algorithm. - * Otherwise, \p modp must point to a function that takes an \p mbedtls_mpi in the - * range of 0..2^(2*pbits)-1, and transforms it in-place to an integer - * which is congruent mod \p P to the given MPI, and is close enough to \p pbits - * in size, so that it may be efficiently brought in the 0..P-1 range by a few - * additions or subtractions. Therefore, it is only an approximative modular - * reduction. It must return 0 on success and non-zero on failure. - * - * \note Alternative implementations of the ECP module must obey the - * following constraints. - * * Group IDs must be distinct: if two group structures have - * the same ID, then they must be identical. - * * The fields \c id, \c P, \c A, \c B, \c G, \c N, - * \c pbits and \c nbits must have the same type and semantics - * as in the built-in implementation. - * They must be available for reading, but direct modification - * of these fields does not need to be supported. - * They do not need to be at the same offset in the structure. - */ -typedef struct mbedtls_ecp_group -{ - mbedtls_ecp_group_id id; /*!< An internal group identifier. */ - mbedtls_mpi P; /*!< The prime modulus of the base field. */ - mbedtls_mpi A; /*!< For Short Weierstrass: \p A in the equation. For - Montgomery curves: (A + 2) / 4. */ - mbedtls_mpi B; /*!< For Short Weierstrass: \p B in the equation. - For Montgomery curves: unused. */ - mbedtls_ecp_point G; /*!< The generator of the subgroup used. */ - mbedtls_mpi N; /*!< The order of \p G. */ - size_t pbits; /*!< The number of bits in \p P.*/ - size_t nbits; /*!< For Short Weierstrass: The number of bits in \p P. - For Montgomery curves: the number of bits in the - private keys. */ - /* End of public fields */ - - unsigned int MBEDTLS_PRIVATE(h); /*!< \internal 1 if the constants are static. */ - int (*MBEDTLS_PRIVATE(modp))(mbedtls_mpi *); /*!< The function for fast pseudo-reduction - mod \p P (see above).*/ - int (*MBEDTLS_PRIVATE(t_pre))(mbedtls_ecp_point *, void *); /*!< Unused. */ - int (*MBEDTLS_PRIVATE(t_post))(mbedtls_ecp_point *, void *); /*!< Unused. */ - void *MBEDTLS_PRIVATE(t_data); /*!< Unused. */ - mbedtls_ecp_point *MBEDTLS_PRIVATE(T); /*!< Pre-computed points for ecp_mul_comb(). */ - size_t MBEDTLS_PRIVATE(T_size); /*!< The number of dynamic allocated pre-computed points. */ -} -mbedtls_ecp_group; - -/** - * \name SECTION: Module settings - * - * The configuration options you can set for this module are in this section. - * Either change them in mbedtls_config.h, or define them using the compiler command line. - * \{ - */ - -#if !defined(MBEDTLS_ECP_WINDOW_SIZE) -/* - * Maximum "window" size used for point multiplication. - * Default: a point where higher memory usage yields disminishing performance - * returns. - * Minimum value: 2. Maximum value: 7. - * - * Result is an array of at most ( 1 << ( MBEDTLS_ECP_WINDOW_SIZE - 1 ) ) - * points used for point multiplication. This value is directly tied to EC - * peak memory usage, so decreasing it by one should roughly cut memory usage - * by two (if large curves are in use). - * - * Reduction in size may reduce speed, but larger curves are impacted first. - * Sample performances (in ECDHE handshakes/s, with FIXED_POINT_OPTIM = 1): - * w-size: 6 5 4 3 2 - * 521 145 141 135 120 97 - * 384 214 209 198 177 146 - * 256 320 320 303 262 226 - * 224 475 475 453 398 342 - * 192 640 640 633 587 476 - */ -#define MBEDTLS_ECP_WINDOW_SIZE 4 /**< The maximum window size used. */ -#endif /* MBEDTLS_ECP_WINDOW_SIZE */ - -#if !defined(MBEDTLS_ECP_FIXED_POINT_OPTIM) -/* - * Trade code size for speed on fixed-point multiplication. - * - * This speeds up repeated multiplication of the generator (that is, the - * multiplication in ECDSA signatures, and half of the multiplications in - * ECDSA verification and ECDHE) by a factor roughly 3 to 4. - * - * For each n-bit Short Weierstrass curve that is enabled, this adds 4n bytes - * of code size if n < 384 and 8n otherwise. - * - * Change this value to 0 to reduce code size. - */ -#define MBEDTLS_ECP_FIXED_POINT_OPTIM 1 /**< Enable fixed-point speed-up. */ -#endif /* MBEDTLS_ECP_FIXED_POINT_OPTIM */ - -/* \} name SECTION: Module settings */ - -#else /* MBEDTLS_ECP_ALT */ -#include "ecp_alt.h" -#endif /* MBEDTLS_ECP_ALT */ - -/** - * The maximum size of the groups, that is, of \c N and \c P. - */ -#if !defined(MBEDTLS_ECP_C) -/* Dummy definition to help code that has optional ECP support and - * defines an MBEDTLS_ECP_MAX_BYTES-sized array unconditionally. */ -#define MBEDTLS_ECP_MAX_BITS 1 -/* Note: the curves must be listed in DECREASING size! */ -#elif defined(MBEDTLS_ECP_DP_SECP521R1_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 521 -#elif defined(MBEDTLS_ECP_DP_BP512R1_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 512 -#elif defined(MBEDTLS_ECP_DP_CURVE448_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 448 -#elif defined(MBEDTLS_ECP_DP_BP384R1_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 384 -#elif defined(MBEDTLS_ECP_DP_SECP384R1_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 384 -#elif defined(MBEDTLS_ECP_DP_BP256R1_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 256 -#elif defined(MBEDTLS_ECP_DP_SECP256K1_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 256 -#elif defined(MBEDTLS_ECP_DP_SECP256R1_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 256 -#elif defined(MBEDTLS_ECP_DP_CURVE25519_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 255 -#elif defined(MBEDTLS_ECP_DP_SECP224K1_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 225 // n is slightly above 2^224 -#elif defined(MBEDTLS_ECP_DP_SECP224R1_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 224 -#elif defined(MBEDTLS_ECP_DP_SECP192K1_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 192 -#elif defined(MBEDTLS_ECP_DP_SECP192R1_ENABLED) -#define MBEDTLS_ECP_MAX_BITS 192 -#else -#error "Missing definition of MBEDTLS_ECP_MAX_BITS" -#endif - -#define MBEDTLS_ECP_MAX_BYTES ( ( MBEDTLS_ECP_MAX_BITS + 7 ) / 8 ) -#define MBEDTLS_ECP_MAX_PT_LEN ( 2 * MBEDTLS_ECP_MAX_BYTES + 1 ) - -#if defined(MBEDTLS_ECP_RESTARTABLE) - -/** - * \brief Internal restart context for multiplication - * - * \note Opaque struct - */ -typedef struct mbedtls_ecp_restart_mul mbedtls_ecp_restart_mul_ctx; - -/** - * \brief Internal restart context for ecp_muladd() - * - * \note Opaque struct - */ -typedef struct mbedtls_ecp_restart_muladd mbedtls_ecp_restart_muladd_ctx; - -/** - * \brief General context for resuming ECC operations - */ -typedef struct -{ - unsigned MBEDTLS_PRIVATE(ops_done); /*!< current ops count */ - unsigned MBEDTLS_PRIVATE(depth); /*!< call depth (0 = top-level) */ - mbedtls_ecp_restart_mul_ctx *MBEDTLS_PRIVATE(rsm); /*!< ecp_mul_comb() sub-context */ - mbedtls_ecp_restart_muladd_ctx *MBEDTLS_PRIVATE(ma); /*!< ecp_muladd() sub-context */ -} mbedtls_ecp_restart_ctx; - -/* - * Operation counts for restartable functions - */ -#define MBEDTLS_ECP_OPS_CHK 3 /*!< basic ops count for ecp_check_pubkey() */ -#define MBEDTLS_ECP_OPS_DBL 8 /*!< basic ops count for ecp_double_jac() */ -#define MBEDTLS_ECP_OPS_ADD 11 /*!< basic ops count for see ecp_add_mixed() */ -#define MBEDTLS_ECP_OPS_INV 120 /*!< empirical equivalent for mpi_mod_inv() */ - -/** - * \brief Internal; for restartable functions in other modules. - * Check and update basic ops budget. - * - * \param grp Group structure - * \param rs_ctx Restart context - * \param ops Number of basic ops to do - * - * \return \c 0 if doing \p ops basic ops is still allowed, - * \return #MBEDTLS_ERR_ECP_IN_PROGRESS otherwise. - */ -int mbedtls_ecp_check_budget( const mbedtls_ecp_group *grp, - mbedtls_ecp_restart_ctx *rs_ctx, - unsigned ops ); - -/* Utility macro for checking and updating ops budget */ -#define MBEDTLS_ECP_BUDGET( ops ) \ - MBEDTLS_MPI_CHK( mbedtls_ecp_check_budget( grp, rs_ctx, \ - (unsigned) (ops) ) ); - -#else /* MBEDTLS_ECP_RESTARTABLE */ - -#define MBEDTLS_ECP_BUDGET( ops ) /* no-op; for compatibility */ - -/* We want to declare restartable versions of existing functions anyway */ -typedef void mbedtls_ecp_restart_ctx; - -#endif /* MBEDTLS_ECP_RESTARTABLE */ - -/** - * \brief The ECP key-pair structure. - * - * A generic key-pair that may be used for ECDSA and fixed ECDH, for example. - * - * \note Members are deliberately in the same order as in the - * ::mbedtls_ecdsa_context structure. - */ -typedef struct mbedtls_ecp_keypair -{ - mbedtls_ecp_group MBEDTLS_PRIVATE(grp); /*!< Elliptic curve and base point */ - mbedtls_mpi MBEDTLS_PRIVATE(d); /*!< our secret value */ - mbedtls_ecp_point MBEDTLS_PRIVATE(Q); /*!< our public value */ -} -mbedtls_ecp_keypair; - -/* - * Point formats, from RFC 4492's enum ECPointFormat - */ -#define MBEDTLS_ECP_PF_UNCOMPRESSED 0 /**< Uncompressed point format. */ -#define MBEDTLS_ECP_PF_COMPRESSED 1 /**< Compressed point format. */ - -/* - * Some other constants from RFC 4492 - */ -#define MBEDTLS_ECP_TLS_NAMED_CURVE 3 /**< The named_curve of ECCurveType. */ - -#if defined(MBEDTLS_ECP_RESTARTABLE) -/** - * \brief Set the maximum number of basic operations done in a row. - * - * If more operations are needed to complete a computation, - * #MBEDTLS_ERR_ECP_IN_PROGRESS will be returned by the - * function performing the computation. It is then the - * caller's responsibility to either call again with the same - * parameters until it returns 0 or an error code; or to free - * the restart context if the operation is to be aborted. - * - * It is strictly required that all input parameters and the - * restart context be the same on successive calls for the - * same operation, but output parameters need not be the - * same; they must not be used until the function finally - * returns 0. - * - * This only applies to functions whose documentation - * mentions they may return #MBEDTLS_ERR_ECP_IN_PROGRESS (or - * #MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS for functions in the - * SSL module). For functions that accept a "restart context" - * argument, passing NULL disables restart and makes the - * function equivalent to the function with the same name - * with \c _restartable removed. For functions in the ECDH - * module, restart is disabled unless the function accepts - * an "ECDH context" argument and - * mbedtls_ecdh_enable_restart() was previously called on - * that context. For function in the SSL module, restart is - * only enabled for specific sides and key exchanges - * (currently only for clients and ECDHE-ECDSA). - * - * \param max_ops Maximum number of basic operations done in a row. - * Default: 0 (unlimited). - * Lower (non-zero) values mean ECC functions will block for - * a lesser maximum amount of time. - * - * \note A "basic operation" is defined as a rough equivalent of a - * multiplication in GF(p) for the NIST P-256 curve. - * As an indication, with default settings, a scalar - * multiplication (full run of \c mbedtls_ecp_mul()) is: - * - about 3300 basic operations for P-256 - * - about 9400 basic operations for P-384 - * - * \note Very low values are not always respected: sometimes - * functions need to block for a minimum number of - * operations, and will do so even if max_ops is set to a - * lower value. That minimum depends on the curve size, and - * can be made lower by decreasing the value of - * \c MBEDTLS_ECP_WINDOW_SIZE. As an indication, here is the - * lowest effective value for various curves and values of - * that parameter (w for short): - * w=6 w=5 w=4 w=3 w=2 - * P-256 208 208 160 136 124 - * P-384 682 416 320 272 248 - * P-521 1364 832 640 544 496 - * - * \note This setting is currently ignored by Curve25519. - */ -void mbedtls_ecp_set_max_ops( unsigned max_ops ); - -/** - * \brief Check if restart is enabled (max_ops != 0) - * - * \return \c 0 if \c max_ops == 0 (restart disabled) - * \return \c 1 otherwise (restart enabled) - */ -int mbedtls_ecp_restart_is_enabled( void ); -#endif /* MBEDTLS_ECP_RESTARTABLE */ - -/* - * Get the type of a curve - */ -mbedtls_ecp_curve_type mbedtls_ecp_get_type( const mbedtls_ecp_group *grp ); - -/** - * \brief This function retrieves the information defined in - * mbedtls_ecp_curve_info() for all supported curves. - * - * \note This function returns information about all curves - * supported by the library. Some curves may not be - * supported for all algorithms. Call mbedtls_ecdh_can_do() - * or mbedtls_ecdsa_can_do() to check if a curve is - * supported for ECDH or ECDSA. - * - * \return A statically allocated array. The last entry is 0. - */ -const mbedtls_ecp_curve_info *mbedtls_ecp_curve_list( void ); - -/** - * \brief This function retrieves the list of internal group - * identifiers of all supported curves in the order of - * preference. - * - * \note This function returns information about all curves - * supported by the library. Some curves may not be - * supported for all algorithms. Call mbedtls_ecdh_can_do() - * or mbedtls_ecdsa_can_do() to check if a curve is - * supported for ECDH or ECDSA. - * - * \return A statically allocated array, - * terminated with MBEDTLS_ECP_DP_NONE. - */ -const mbedtls_ecp_group_id *mbedtls_ecp_grp_id_list( void ); - -/** - * \brief This function retrieves curve information from an internal - * group identifier. - * - * \param grp_id An \c MBEDTLS_ECP_DP_XXX value. - * - * \return The associated curve information on success. - * \return NULL on failure. - */ -const mbedtls_ecp_curve_info *mbedtls_ecp_curve_info_from_grp_id( mbedtls_ecp_group_id grp_id ); - -/** - * \brief This function retrieves curve information from a TLS - * NamedCurve value. - * - * \param tls_id An \c MBEDTLS_ECP_DP_XXX value. - * - * \return The associated curve information on success. - * \return NULL on failure. - */ -const mbedtls_ecp_curve_info *mbedtls_ecp_curve_info_from_tls_id( uint16_t tls_id ); - -/** - * \brief This function retrieves curve information from a - * human-readable name. - * - * \param name The human-readable name. - * - * \return The associated curve information on success. - * \return NULL on failure. - */ -const mbedtls_ecp_curve_info *mbedtls_ecp_curve_info_from_name( const char *name ); - -/** - * \brief This function initializes a point as zero. - * - * \param pt The point to initialize. - */ -void mbedtls_ecp_point_init( mbedtls_ecp_point *pt ); - -/** - * \brief This function initializes an ECP group context - * without loading any domain parameters. - * - * \note After this function is called, domain parameters - * for various ECP groups can be loaded through the - * mbedtls_ecp_group_load() or mbedtls_ecp_tls_read_group() - * functions. - */ -void mbedtls_ecp_group_init( mbedtls_ecp_group *grp ); - -/** - * \brief This function initializes a key pair as an invalid one. - * - * \param key The key pair to initialize. - */ -void mbedtls_ecp_keypair_init( mbedtls_ecp_keypair *key ); - -/** - * \brief This function frees the components of a point. - * - * \param pt The point to free. - */ -void mbedtls_ecp_point_free( mbedtls_ecp_point *pt ); - -/** - * \brief This function frees the components of an ECP group. - * - * \param grp The group to free. This may be \c NULL, in which - * case this function returns immediately. If it is not - * \c NULL, it must point to an initialized ECP group. - */ -void mbedtls_ecp_group_free( mbedtls_ecp_group *grp ); - -/** - * \brief This function frees the components of a key pair. - * - * \param key The key pair to free. This may be \c NULL, in which - * case this function returns immediately. If it is not - * \c NULL, it must point to an initialized ECP key pair. - */ -void mbedtls_ecp_keypair_free( mbedtls_ecp_keypair *key ); - -#if defined(MBEDTLS_ECP_RESTARTABLE) -/** - * \brief Initialize a restart context. - * - * \param ctx The restart context to initialize. This must - * not be \c NULL. - */ -void mbedtls_ecp_restart_init( mbedtls_ecp_restart_ctx *ctx ); - -/** - * \brief Free the components of a restart context. - * - * \param ctx The restart context to free. This may be \c NULL, in which - * case this function returns immediately. If it is not - * \c NULL, it must point to an initialized restart context. - */ -void mbedtls_ecp_restart_free( mbedtls_ecp_restart_ctx *ctx ); -#endif /* MBEDTLS_ECP_RESTARTABLE */ - -/** - * \brief This function copies the contents of point \p Q into - * point \p P. - * - * \param P The destination point. This must be initialized. - * \param Q The source point. This must be initialized. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED on memory-allocation failure. - * \return Another negative error code for other kinds of failure. - */ -int mbedtls_ecp_copy( mbedtls_ecp_point *P, const mbedtls_ecp_point *Q ); - -/** - * \brief This function copies the contents of group \p src into - * group \p dst. - * - * \param dst The destination group. This must be initialized. - * \param src The source group. This must be initialized. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED on memory-allocation failure. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_group_copy( mbedtls_ecp_group *dst, - const mbedtls_ecp_group *src ); - -/** - * \brief This function sets a point to the point at infinity. - * - * \param pt The point to set. This must be initialized. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED on memory-allocation failure. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_set_zero( mbedtls_ecp_point *pt ); - -/** - * \brief This function checks if a point is the point at infinity. - * - * \param pt The point to test. This must be initialized. - * - * \return \c 1 if the point is zero. - * \return \c 0 if the point is non-zero. - * \return A negative error code on failure. - */ -int mbedtls_ecp_is_zero( mbedtls_ecp_point *pt ); - -/** - * \brief This function compares two points. - * - * \note This assumes that the points are normalized. Otherwise, - * they may compare as "not equal" even if they are. - * - * \param P The first point to compare. This must be initialized. - * \param Q The second point to compare. This must be initialized. - * - * \return \c 0 if the points are equal. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA if the points are not equal. - */ -int mbedtls_ecp_point_cmp( const mbedtls_ecp_point *P, - const mbedtls_ecp_point *Q ); - -/** - * \brief This function imports a non-zero point from two ASCII - * strings. - * - * \param P The destination point. This must be initialized. - * \param radix The numeric base of the input. - * \param x The first affine coordinate, as a null-terminated string. - * \param y The second affine coordinate, as a null-terminated string. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_MPI_XXX error code on failure. - */ -int mbedtls_ecp_point_read_string( mbedtls_ecp_point *P, int radix, - const char *x, const char *y ); - -/** - * \brief This function exports a point into unsigned binary data. - * - * \param grp The group to which the point should belong. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param P The point to export. This must be initialized. - * \param format The point format. This must be either - * #MBEDTLS_ECP_PF_COMPRESSED or #MBEDTLS_ECP_PF_UNCOMPRESSED. - * (For groups without these formats, this parameter is - * ignored. But it still has to be either of the above - * values.) - * \param olen The address at which to store the length of - * the output in Bytes. This must not be \c NULL. - * \param buf The output buffer. This must be a writable buffer - * of length \p buflen Bytes. - * \param buflen The length of the output buffer \p buf in Bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_BUFFER_TOO_SMALL if the output buffer - * is too small to hold the point. - * \return #MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE if the point format - * or the export for the given group is not implemented. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_point_write_binary( const mbedtls_ecp_group *grp, - const mbedtls_ecp_point *P, - int format, size_t *olen, - unsigned char *buf, size_t buflen ); - -/** - * \brief This function imports a point from unsigned binary data. - * - * \note This function does not check that the point actually - * belongs to the given group, see mbedtls_ecp_check_pubkey() - * for that. - * - * \param grp The group to which the point should belong. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param P The destination context to import the point to. - * This must be initialized. - * \param buf The input buffer. This must be a readable buffer - * of length \p ilen Bytes. - * \param ilen The length of the input buffer \p buf in Bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA if the input is invalid. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED on memory-allocation failure. - * \return #MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE if the import for the - * given group is not implemented. - */ -int mbedtls_ecp_point_read_binary( const mbedtls_ecp_group *grp, - mbedtls_ecp_point *P, - const unsigned char *buf, size_t ilen ); - -/** - * \brief This function imports a point from a TLS ECPoint record. - * - * \note On function return, \p *buf is updated to point immediately - * after the ECPoint record. - * - * \param grp The ECP group to use. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param pt The destination point. - * \param buf The address of the pointer to the start of the input buffer. - * \param len The length of the buffer. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_MPI_XXX error code on initialization - * failure. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA if input is invalid. - */ -int mbedtls_ecp_tls_read_point( const mbedtls_ecp_group *grp, - mbedtls_ecp_point *pt, - const unsigned char **buf, size_t len ); - -/** - * \brief This function exports a point as a TLS ECPoint record - * defined in RFC 4492, Section 5.4. - * - * \param grp The ECP group to use. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param pt The point to be exported. This must be initialized. - * \param format The point format to use. This must be either - * #MBEDTLS_ECP_PF_COMPRESSED or #MBEDTLS_ECP_PF_UNCOMPRESSED. - * \param olen The address at which to store the length in Bytes - * of the data written. - * \param buf The target buffer. This must be a writable buffer of - * length \p blen Bytes. - * \param blen The length of the target buffer \p buf in Bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA if the input is invalid. - * \return #MBEDTLS_ERR_ECP_BUFFER_TOO_SMALL if the target buffer - * is too small to hold the exported point. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_tls_write_point( const mbedtls_ecp_group *grp, - const mbedtls_ecp_point *pt, - int format, size_t *olen, - unsigned char *buf, size_t blen ); - -/** - * \brief This function sets up an ECP group context - * from a standardized set of domain parameters. - * - * \note The index should be a value of the NamedCurve enum, - * as defined in RFC-4492: Elliptic Curve Cryptography - * (ECC) Cipher Suites for Transport Layer Security (TLS), - * usually in the form of an \c MBEDTLS_ECP_DP_XXX macro. - * - * \param grp The group context to setup. This must be initialized. - * \param id The identifier of the domain parameter set to load. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE if \p id doesn't - * correspond to a known group. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_group_load( mbedtls_ecp_group *grp, mbedtls_ecp_group_id id ); - -/** - * \brief This function sets up an ECP group context from a TLS - * ECParameters record as defined in RFC 4492, Section 5.4. - * - * \note The read pointer \p buf is updated to point right after - * the ECParameters record on exit. - * - * \param grp The group context to setup. This must be initialized. - * \param buf The address of the pointer to the start of the input buffer. - * \param len The length of the input buffer \c *buf in Bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA if input is invalid. - * \return #MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE if the group is not - * recognized. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_tls_read_group( mbedtls_ecp_group *grp, - const unsigned char **buf, size_t len ); - -/** - * \brief This function extracts an elliptic curve group ID from a - * TLS ECParameters record as defined in RFC 4492, Section 5.4. - * - * \note The read pointer \p buf is updated to point right after - * the ECParameters record on exit. - * - * \param grp The address at which to store the group id. - * This must not be \c NULL. - * \param buf The address of the pointer to the start of the input buffer. - * \param len The length of the input buffer \c *buf in Bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA if input is invalid. - * \return #MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE if the group is not - * recognized. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_tls_read_group_id( mbedtls_ecp_group_id *grp, - const unsigned char **buf, - size_t len ); -/** - * \brief This function exports an elliptic curve as a TLS - * ECParameters record as defined in RFC 4492, Section 5.4. - * - * \param grp The ECP group to be exported. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param olen The address at which to store the number of Bytes written. - * This must not be \c NULL. - * \param buf The buffer to write to. This must be a writable buffer - * of length \p blen Bytes. - * \param blen The length of the output buffer \p buf in Bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_BUFFER_TOO_SMALL if the output - * buffer is too small to hold the exported group. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_tls_write_group( const mbedtls_ecp_group *grp, - size_t *olen, - unsigned char *buf, size_t blen ); - -/** - * \brief This function performs a scalar multiplication of a point - * by an integer: \p R = \p m * \p P. - * - * It is not thread-safe to use same group in multiple threads. - * - * \note To prevent timing attacks, this function - * executes the exact same sequence of base-field - * operations for any valid \p m. It avoids any if-branch or - * array index depending on the value of \p m. If also uses - * \p f_rng to randomize some intermediate results. - * - * \param grp The ECP group to use. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param R The point in which to store the result of the calculation. - * This must be initialized. - * \param m The integer by which to multiply. This must be initialized. - * \param P The point to multiply. This must be initialized. - * \param f_rng The RNG function. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be \c - * NULL if \p f_rng doesn't need a context. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_INVALID_KEY if \p m is not a valid private - * key, or \p P is not a valid public key. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED on memory-allocation failure. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_mul( mbedtls_ecp_group *grp, mbedtls_ecp_point *R, - const mbedtls_mpi *m, const mbedtls_ecp_point *P, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng ); - -/** - * \brief This function performs multiplication of a point by - * an integer: \p R = \p m * \p P in a restartable way. - * - * \see mbedtls_ecp_mul() - * - * \note This function does the same as \c mbedtls_ecp_mul(), but - * it can return early and restart according to the limit set - * with \c mbedtls_ecp_set_max_ops() to reduce blocking. - * - * \param grp The ECP group to use. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param R The point in which to store the result of the calculation. - * This must be initialized. - * \param m The integer by which to multiply. This must be initialized. - * \param P The point to multiply. This must be initialized. - * \param f_rng The RNG function. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be \c - * NULL if \p f_rng doesn't need a context. - * \param rs_ctx The restart context (NULL disables restart). - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_INVALID_KEY if \p m is not a valid private - * key, or \p P is not a valid public key. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED on memory-allocation failure. - * \return #MBEDTLS_ERR_ECP_IN_PROGRESS if maximum number of - * operations was reached: see \c mbedtls_ecp_set_max_ops(). - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_mul_restartable( mbedtls_ecp_group *grp, mbedtls_ecp_point *R, - const mbedtls_mpi *m, const mbedtls_ecp_point *P, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng, - mbedtls_ecp_restart_ctx *rs_ctx ); - -#if defined(MBEDTLS_ECP_SHORT_WEIERSTRASS_ENABLED) -/** - * \brief This function performs multiplication and addition of two - * points by integers: \p R = \p m * \p P + \p n * \p Q - * - * It is not thread-safe to use same group in multiple threads. - * - * \note In contrast to mbedtls_ecp_mul(), this function does not - * guarantee a constant execution flow and timing. - * - * \note This function is only defined for short Weierstrass curves. - * It may not be included in builds without any short - * Weierstrass curve. - * - * \param grp The ECP group to use. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param R The point in which to store the result of the calculation. - * This must be initialized. - * \param m The integer by which to multiply \p P. - * This must be initialized. - * \param P The point to multiply by \p m. This must be initialized. - * \param n The integer by which to multiply \p Q. - * This must be initialized. - * \param Q The point to be multiplied by \p n. - * This must be initialized. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_INVALID_KEY if \p m or \p n are not - * valid private keys, or \p P or \p Q are not valid public - * keys. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED on memory-allocation failure. - * \return #MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE if \p grp does not - * designate a short Weierstrass curve. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_muladd( mbedtls_ecp_group *grp, mbedtls_ecp_point *R, - const mbedtls_mpi *m, const mbedtls_ecp_point *P, - const mbedtls_mpi *n, const mbedtls_ecp_point *Q ); - -/** - * \brief This function performs multiplication and addition of two - * points by integers: \p R = \p m * \p P + \p n * \p Q in a - * restartable way. - * - * \see \c mbedtls_ecp_muladd() - * - * \note This function works the same as \c mbedtls_ecp_muladd(), - * but it can return early and restart according to the limit - * set with \c mbedtls_ecp_set_max_ops() to reduce blocking. - * - * \note This function is only defined for short Weierstrass curves. - * It may not be included in builds without any short - * Weierstrass curve. - * - * \param grp The ECP group to use. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param R The point in which to store the result of the calculation. - * This must be initialized. - * \param m The integer by which to multiply \p P. - * This must be initialized. - * \param P The point to multiply by \p m. This must be initialized. - * \param n The integer by which to multiply \p Q. - * This must be initialized. - * \param Q The point to be multiplied by \p n. - * This must be initialized. - * \param rs_ctx The restart context (NULL disables restart). - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_INVALID_KEY if \p m or \p n are not - * valid private keys, or \p P or \p Q are not valid public - * keys. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED on memory-allocation failure. - * \return #MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE if \p grp does not - * designate a short Weierstrass curve. - * \return #MBEDTLS_ERR_ECP_IN_PROGRESS if maximum number of - * operations was reached: see \c mbedtls_ecp_set_max_ops(). - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_muladd_restartable( - mbedtls_ecp_group *grp, mbedtls_ecp_point *R, - const mbedtls_mpi *m, const mbedtls_ecp_point *P, - const mbedtls_mpi *n, const mbedtls_ecp_point *Q, - mbedtls_ecp_restart_ctx *rs_ctx ); -#endif /* MBEDTLS_ECP_SHORT_WEIERSTRASS_ENABLED */ - -/** - * \brief This function checks that a point is a valid public key - * on this curve. - * - * It only checks that the point is non-zero, has - * valid coordinates and lies on the curve. It does not verify - * that it is indeed a multiple of \p G. This additional - * check is computationally more expensive, is not required - * by standards, and should not be necessary if the group - * used has a small cofactor. In particular, it is useless for - * the NIST groups which all have a cofactor of 1. - * - * \note This function uses bare components rather than an - * ::mbedtls_ecp_keypair structure, to ease use with other - * structures, such as ::mbedtls_ecdh_context or - * ::mbedtls_ecdsa_context. - * - * \param grp The ECP group the point should belong to. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param pt The point to check. This must be initialized. - * - * \return \c 0 if the point is a valid public key. - * \return #MBEDTLS_ERR_ECP_INVALID_KEY if the point is not - * a valid public key for the given curve. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_check_pubkey( const mbedtls_ecp_group *grp, - const mbedtls_ecp_point *pt ); - -/** - * \brief This function checks that an \p mbedtls_mpi is a - * valid private key for this curve. - * - * \note This function uses bare components rather than an - * ::mbedtls_ecp_keypair structure to ease use with other - * structures, such as ::mbedtls_ecdh_context or - * ::mbedtls_ecdsa_context. - * - * \param grp The ECP group the private key should belong to. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param d The integer to check. This must be initialized. - * - * \return \c 0 if the point is a valid private key. - * \return #MBEDTLS_ERR_ECP_INVALID_KEY if the point is not a valid - * private key for the given curve. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ecp_check_privkey( const mbedtls_ecp_group *grp, - const mbedtls_mpi *d ); - -/** - * \brief This function generates a private key. - * - * \param grp The ECP group to generate a private key for. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param d The destination MPI (secret part). This must be initialized. - * \param f_rng The RNG function. This must not be \c NULL. - * \param p_rng The RNG parameter to be passed to \p f_rng. This may be - * \c NULL if \p f_rng doesn't need a context argument. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_ECP_XXX or \c MBEDTLS_MPI_XXX error code - * on failure. - */ -int mbedtls_ecp_gen_privkey( const mbedtls_ecp_group *grp, - mbedtls_mpi *d, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function generates a keypair with a configurable base - * point. - * - * \note This function uses bare components rather than an - * ::mbedtls_ecp_keypair structure to ease use with other - * structures, such as ::mbedtls_ecdh_context or - * ::mbedtls_ecdsa_context. - * - * \param grp The ECP group to generate a key pair for. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param G The base point to use. This must be initialized - * and belong to \p grp. It replaces the default base - * point \c grp->G used by mbedtls_ecp_gen_keypair(). - * \param d The destination MPI (secret part). - * This must be initialized. - * \param Q The destination point (public part). - * This must be initialized. - * \param f_rng The RNG function. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may - * be \c NULL if \p f_rng doesn't need a context argument. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_ECP_XXX or \c MBEDTLS_MPI_XXX error code - * on failure. - */ -int mbedtls_ecp_gen_keypair_base( mbedtls_ecp_group *grp, - const mbedtls_ecp_point *G, - mbedtls_mpi *d, mbedtls_ecp_point *Q, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function generates an ECP keypair. - * - * \note This function uses bare components rather than an - * ::mbedtls_ecp_keypair structure to ease use with other - * structures, such as ::mbedtls_ecdh_context or - * ::mbedtls_ecdsa_context. - * - * \param grp The ECP group to generate a key pair for. - * This must be initialized and have group parameters - * set, for example through mbedtls_ecp_group_load(). - * \param d The destination MPI (secret part). - * This must be initialized. - * \param Q The destination point (public part). - * This must be initialized. - * \param f_rng The RNG function. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may - * be \c NULL if \p f_rng doesn't need a context argument. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_ECP_XXX or \c MBEDTLS_MPI_XXX error code - * on failure. - */ -int mbedtls_ecp_gen_keypair( mbedtls_ecp_group *grp, mbedtls_mpi *d, - mbedtls_ecp_point *Q, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function generates an ECP key. - * - * \param grp_id The ECP group identifier. - * \param key The destination key. This must be initialized. - * \param f_rng The RNG function to use. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may - * be \c NULL if \p f_rng doesn't need a context argument. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_ECP_XXX or \c MBEDTLS_MPI_XXX error code - * on failure. - */ -int mbedtls_ecp_gen_key( mbedtls_ecp_group_id grp_id, mbedtls_ecp_keypair *key, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief This function reads an elliptic curve private key. - * - * \param grp_id The ECP group identifier. - * \param key The destination key. - * \param buf The buffer containing the binary representation of the - * key. (Big endian integer for Weierstrass curves, byte - * string for Montgomery curves.) - * \param buflen The length of the buffer in bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_INVALID_KEY error if the key is - * invalid. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED if memory allocation failed. - * \return #MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE if the operation for - * the group is not implemented. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_ecp_read_key( mbedtls_ecp_group_id grp_id, mbedtls_ecp_keypair *key, - const unsigned char *buf, size_t buflen ); - -/** - * \brief This function exports an elliptic curve private key. - * - * \param key The private key. - * \param buf The output buffer for containing the binary representation - * of the key. (Big endian integer for Weierstrass curves, byte - * string for Montgomery curves.) - * \param buflen The total length of the buffer in bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_ECP_BUFFER_TOO_SMALL if the \p key - representation is larger than the available space in \p buf. - * \return #MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE if the operation for - * the group is not implemented. - * \return Another negative error code on different kinds of failure. - */ -int mbedtls_ecp_write_key( mbedtls_ecp_keypair *key, - unsigned char *buf, size_t buflen ); - -/** - * \brief This function checks that the keypair objects - * \p pub and \p prv have the same group and the - * same public point, and that the private key in - * \p prv is consistent with the public key. - * - * \param pub The keypair structure holding the public key. This - * must be initialized. If it contains a private key, that - * part is ignored. - * \param prv The keypair structure holding the full keypair. - * This must be initialized. - * \param f_rng The RNG function. This must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be \c - * NULL if \p f_rng doesn't need a context. - * - * \return \c 0 on success, meaning that the keys are valid and match. - * \return #MBEDTLS_ERR_ECP_BAD_INPUT_DATA if the keys are invalid or do not match. - * \return An \c MBEDTLS_ERR_ECP_XXX or an \c MBEDTLS_ERR_MPI_XXX - * error code on calculation failure. - */ -int mbedtls_ecp_check_pub_priv( - const mbedtls_ecp_keypair *pub, const mbedtls_ecp_keypair *prv, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng ); - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief The ECP checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_ecp_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* ecp.h */ diff --git a/windows/mbedtls/entropy.h b/windows/mbedtls/entropy.h deleted file mode 100644 index a16a3753..00000000 --- a/windows/mbedtls/entropy.h +++ /dev/null @@ -1,288 +0,0 @@ -/** - * \file entropy.h - * - * \brief Entropy accumulator implementation - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_ENTROPY_H -#define MBEDTLS_ENTROPY_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include - -#if defined(MBEDTLS_SHA512_C) && !defined(MBEDTLS_ENTROPY_FORCE_SHA256) -#include "mbedtls/sha512.h" -#define MBEDTLS_ENTROPY_SHA512_ACCUMULATOR -#else -#if defined(MBEDTLS_SHA256_C) -#define MBEDTLS_ENTROPY_SHA256_ACCUMULATOR -#include "mbedtls/sha256.h" -#endif -#endif - -#if defined(MBEDTLS_THREADING_C) -#include "mbedtls/threading.h" -#endif - - -#define MBEDTLS_ERR_ENTROPY_SOURCE_FAILED -0x003C /**< Critical entropy source failure. */ -#define MBEDTLS_ERR_ENTROPY_MAX_SOURCES -0x003E /**< No more sources can be added. */ -#define MBEDTLS_ERR_ENTROPY_NO_SOURCES_DEFINED -0x0040 /**< No sources have been added to poll. */ -#define MBEDTLS_ERR_ENTROPY_NO_STRONG_SOURCE -0x003D /**< No strong sources have been added to poll. */ -#define MBEDTLS_ERR_ENTROPY_FILE_IO_ERROR -0x003F /**< Read/write error in file. */ - -/** - * \name SECTION: Module settings - * - * The configuration options you can set for this module are in this section. - * Either change them in mbedtls_config.h or define them on the compiler command line. - * \{ - */ - -#if !defined(MBEDTLS_ENTROPY_MAX_SOURCES) -#define MBEDTLS_ENTROPY_MAX_SOURCES 20 /**< Maximum number of sources supported */ -#endif - -#if !defined(MBEDTLS_ENTROPY_MAX_GATHER) -#define MBEDTLS_ENTROPY_MAX_GATHER 128 /**< Maximum amount requested from entropy sources */ -#endif - -/* \} name SECTION: Module settings */ - -#if defined(MBEDTLS_ENTROPY_SHA512_ACCUMULATOR) -#define MBEDTLS_ENTROPY_BLOCK_SIZE 64 /**< Block size of entropy accumulator (SHA-512) */ -#else -#define MBEDTLS_ENTROPY_BLOCK_SIZE 32 /**< Block size of entropy accumulator (SHA-256) */ -#endif - -#define MBEDTLS_ENTROPY_MAX_SEED_SIZE 1024 /**< Maximum size of seed we read from seed file */ -#define MBEDTLS_ENTROPY_SOURCE_MANUAL MBEDTLS_ENTROPY_MAX_SOURCES - -#define MBEDTLS_ENTROPY_SOURCE_STRONG 1 /**< Entropy source is strong */ -#define MBEDTLS_ENTROPY_SOURCE_WEAK 0 /**< Entropy source is weak */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief Entropy poll callback pointer - * - * \param data Callback-specific data pointer - * \param output Data to fill - * \param len Maximum size to provide - * \param olen The actual amount of bytes put into the buffer (Can be 0) - * - * \return 0 if no critical failures occurred, - * MBEDTLS_ERR_ENTROPY_SOURCE_FAILED otherwise - */ -typedef int (*mbedtls_entropy_f_source_ptr)(void *data, unsigned char *output, size_t len, - size_t *olen); - -/** - * \brief Entropy source state - */ -typedef struct mbedtls_entropy_source_state -{ - mbedtls_entropy_f_source_ptr MBEDTLS_PRIVATE(f_source); /**< The entropy source callback */ - void * MBEDTLS_PRIVATE(p_source); /**< The callback data pointer */ - size_t MBEDTLS_PRIVATE(size); /**< Amount received in bytes */ - size_t MBEDTLS_PRIVATE(threshold); /**< Minimum bytes required before release */ - int MBEDTLS_PRIVATE(strong); /**< Is the source strong? */ -} -mbedtls_entropy_source_state; - -/** - * \brief Entropy context structure - */ -typedef struct mbedtls_entropy_context -{ - int MBEDTLS_PRIVATE(accumulator_started); /* 0 after init. - * 1 after the first update. - * -1 after free. */ -#if defined(MBEDTLS_ENTROPY_SHA512_ACCUMULATOR) - mbedtls_sha512_context MBEDTLS_PRIVATE(accumulator); -#else - mbedtls_sha256_context MBEDTLS_PRIVATE(accumulator); -#endif - int MBEDTLS_PRIVATE(source_count); /* Number of entries used in source. */ - mbedtls_entropy_source_state MBEDTLS_PRIVATE(source)[MBEDTLS_ENTROPY_MAX_SOURCES]; -#if defined(MBEDTLS_THREADING_C) - mbedtls_threading_mutex_t MBEDTLS_PRIVATE(mutex); /*!< mutex */ -#endif -#if defined(MBEDTLS_ENTROPY_NV_SEED) - int MBEDTLS_PRIVATE(initial_entropy_run); -#endif -} -mbedtls_entropy_context; - -#if !defined(MBEDTLS_NO_PLATFORM_ENTROPY) -/** - * \brief Platform-specific entropy poll callback - */ -int mbedtls_platform_entropy_poll( void *data, - unsigned char *output, size_t len, size_t *olen ); -#endif - -/** - * \brief Initialize the context - * - * \param ctx Entropy context to initialize - */ -void mbedtls_entropy_init( mbedtls_entropy_context *ctx ); - -/** - * \brief Free the data in the context - * - * \param ctx Entropy context to free - */ -void mbedtls_entropy_free( mbedtls_entropy_context *ctx ); - -/** - * \brief Adds an entropy source to poll - * (Thread-safe if MBEDTLS_THREADING_C is enabled) - * - * \param ctx Entropy context - * \param f_source Entropy function - * \param p_source Function data - * \param threshold Minimum required from source before entropy is released - * ( with mbedtls_entropy_func() ) (in bytes) - * \param strong MBEDTLS_ENTROPY_SOURCE_STRONG or - * MBEDTLS_ENTROPY_SOURCE_WEAK. - * At least one strong source needs to be added. - * Weaker sources (such as the cycle counter) can be used as - * a complement. - * - * \return 0 if successful or MBEDTLS_ERR_ENTROPY_MAX_SOURCES - */ -int mbedtls_entropy_add_source( mbedtls_entropy_context *ctx, - mbedtls_entropy_f_source_ptr f_source, void *p_source, - size_t threshold, int strong ); - -/** - * \brief Trigger an extra gather poll for the accumulator - * (Thread-safe if MBEDTLS_THREADING_C is enabled) - * - * \param ctx Entropy context - * - * \return 0 if successful, or MBEDTLS_ERR_ENTROPY_SOURCE_FAILED - */ -int mbedtls_entropy_gather( mbedtls_entropy_context *ctx ); - -/** - * \brief Retrieve entropy from the accumulator - * (Maximum length: MBEDTLS_ENTROPY_BLOCK_SIZE) - * (Thread-safe if MBEDTLS_THREADING_C is enabled) - * - * \param data Entropy context - * \param output Buffer to fill - * \param len Number of bytes desired, must be at most MBEDTLS_ENTROPY_BLOCK_SIZE - * - * \return 0 if successful, or MBEDTLS_ERR_ENTROPY_SOURCE_FAILED - */ -int mbedtls_entropy_func( void *data, unsigned char *output, size_t len ); - -/** - * \brief Add data to the accumulator manually - * (Thread-safe if MBEDTLS_THREADING_C is enabled) - * - * \param ctx Entropy context - * \param data Data to add - * \param len Length of data - * - * \return 0 if successful - */ -int mbedtls_entropy_update_manual( mbedtls_entropy_context *ctx, - const unsigned char *data, size_t len ); - -#if defined(MBEDTLS_ENTROPY_NV_SEED) -/** - * \brief Trigger an update of the seed file in NV by using the - * current entropy pool. - * - * \param ctx Entropy context - * - * \return 0 if successful - */ -int mbedtls_entropy_update_nv_seed( mbedtls_entropy_context *ctx ); -#endif /* MBEDTLS_ENTROPY_NV_SEED */ - -#if defined(MBEDTLS_FS_IO) -/** - * \brief Write a seed file - * - * \param ctx Entropy context - * \param path Name of the file - * - * \return 0 if successful, - * MBEDTLS_ERR_ENTROPY_FILE_IO_ERROR on file error, or - * MBEDTLS_ERR_ENTROPY_SOURCE_FAILED - */ -int mbedtls_entropy_write_seed_file( mbedtls_entropy_context *ctx, const char *path ); - -/** - * \brief Read and update a seed file. Seed is added to this - * instance. No more than MBEDTLS_ENTROPY_MAX_SEED_SIZE bytes are - * read from the seed file. The rest is ignored. - * - * \param ctx Entropy context - * \param path Name of the file - * - * \return 0 if successful, - * MBEDTLS_ERR_ENTROPY_FILE_IO_ERROR on file error, - * MBEDTLS_ERR_ENTROPY_SOURCE_FAILED - */ -int mbedtls_entropy_update_seed_file( mbedtls_entropy_context *ctx, const char *path ); -#endif /* MBEDTLS_FS_IO */ - -#if defined(MBEDTLS_SELF_TEST) -/** - * \brief Checkup routine - * - * This module self-test also calls the entropy self-test, - * mbedtls_entropy_source_self_test(); - * - * \return 0 if successful, or 1 if a test failed - */ -int mbedtls_entropy_self_test( int verbose ); - -#if defined(MBEDTLS_ENTROPY_HARDWARE_ALT) -/** - * \brief Checkup routine - * - * Verifies the integrity of the hardware entropy source - * provided by the function 'mbedtls_hardware_poll()'. - * - * Note this is the only hardware entropy source that is known - * at link time, and other entropy sources configured - * dynamically at runtime by the function - * mbedtls_entropy_add_source() will not be tested. - * - * \return 0 if successful, or 1 if a test failed - */ -int mbedtls_entropy_source_self_test( int verbose ); -#endif /* MBEDTLS_ENTROPY_HARDWARE_ALT */ -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* entropy.h */ diff --git a/windows/mbedtls/error.h b/windows/mbedtls/error.h deleted file mode 100644 index 9a8690dc..00000000 --- a/windows/mbedtls/error.h +++ /dev/null @@ -1,210 +0,0 @@ -/** - * \file error.h - * - * \brief Error to string translation - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_ERROR_H -#define MBEDTLS_ERROR_H - -#include "mbedtls/build_info.h" - -#include - -#if ( defined(__ARMCC_VERSION) || defined(_MSC_VER) ) && \ - !defined(inline) && !defined(__cplusplus) -#define inline __inline -#endif - -/** - * Error code layout. - * - * Currently we try to keep all error codes within the negative space of 16 - * bits signed integers to support all platforms (-0x0001 - -0x7FFF). In - * addition we'd like to give two layers of information on the error if - * possible. - * - * For that purpose the error codes are segmented in the following manner: - * - * 16 bit error code bit-segmentation - * - * 1 bit - Unused (sign bit) - * 3 bits - High level module ID - * 5 bits - Module-dependent error code - * 7 bits - Low level module errors - * - * For historical reasons, low-level error codes are divided in even and odd, - * even codes were assigned first, and -1 is reserved for other errors. - * - * Low-level module errors (0x0002-0x007E, 0x0001-0x007F) - * - * Module Nr Codes assigned - * ERROR 2 0x006E 0x0001 - * MPI 7 0x0002-0x0010 - * GCM 3 0x0012-0x0014 0x0013-0x0013 - * THREADING 3 0x001A-0x001E - * AES 5 0x0020-0x0022 0x0021-0x0025 - * CAMELLIA 3 0x0024-0x0026 0x0027-0x0027 - * BASE64 2 0x002A-0x002C - * OID 1 0x002E-0x002E 0x000B-0x000B - * PADLOCK 1 0x0030-0x0030 - * DES 2 0x0032-0x0032 0x0033-0x0033 - * CTR_DBRG 4 0x0034-0x003A - * ENTROPY 3 0x003C-0x0040 0x003D-0x003F - * NET 13 0x0042-0x0052 0x0043-0x0049 - * ARIA 4 0x0058-0x005E - * ASN1 7 0x0060-0x006C - * CMAC 1 0x007A-0x007A - * PBKDF2 1 0x007C-0x007C - * HMAC_DRBG 4 0x0003-0x0009 - * CCM 3 0x000D-0x0011 - * MD5 1 0x002F-0x002F - * RIPEMD160 1 0x0031-0x0031 - * SHA1 1 0x0035-0x0035 0x0073-0x0073 - * SHA256 1 0x0037-0x0037 0x0074-0x0074 - * SHA512 1 0x0039-0x0039 0x0075-0x0075 - * CHACHA20 3 0x0051-0x0055 - * POLY1305 3 0x0057-0x005B - * CHACHAPOLY 2 0x0054-0x0056 - * PLATFORM 2 0x0070-0x0072 - * - * High-level module nr (3 bits - 0x0...-0x7...) - * Name ID Nr of Errors - * PEM 1 9 - * PKCS#12 1 4 (Started from top) - * X509 2 20 - * PKCS5 2 4 (Started from top) - * DHM 3 11 - * PK 3 15 (Started from top) - * RSA 4 11 - * ECP 4 10 (Started from top) - * MD 5 5 - * HKDF 5 1 (Started from top) - * SSL 5 2 (Started from 0x5F00) - * CIPHER 6 8 (Started from 0x6080) - * SSL 6 22 (Started from top, plus 0x6000) - * SSL 7 20 (Started from 0x7000, gaps at - * 0x7380, 0x7900-0x7980, 0x7A80-0x7E80) - * - * Module dependent error code (5 bits 0x.00.-0x.F8.) - */ - -#ifdef __cplusplus -extern "C" { -#endif - -#define MBEDTLS_ERR_ERROR_GENERIC_ERROR -0x0001 /**< Generic error */ -#define MBEDTLS_ERR_ERROR_CORRUPTION_DETECTED -0x006E /**< This is a bug in the library */ - -#define MBEDTLS_ERR_PLATFORM_HW_ACCEL_FAILED -0x0070 /**< Hardware accelerator failed */ -#define MBEDTLS_ERR_PLATFORM_FEATURE_UNSUPPORTED -0x0072 /**< The requested feature is not supported by the platform */ - -/** - * \brief Combines a high-level and low-level error code together. - * - * Wrapper macro for mbedtls_error_add(). See that function for - * more details. - */ -#define MBEDTLS_ERROR_ADD( high, low ) \ - mbedtls_error_add( high, low, __FILE__, __LINE__ ) - -#if defined(MBEDTLS_TEST_HOOKS) -/** - * \brief Testing hook called before adding/combining two error codes together. - * Only used when invasive testing is enabled via MBEDTLS_TEST_HOOKS. - */ -extern void (*mbedtls_test_hook_error_add)( int, int, const char *, int ); -#endif - -/** - * \brief Combines a high-level and low-level error code together. - * - * This function can be called directly however it is usually - * called via the #MBEDTLS_ERROR_ADD macro. - * - * While a value of zero is not a negative error code, it is still an - * error code (that denotes success) and can be combined with both a - * negative error code or another value of zero. - * - * \note When invasive testing is enabled via #MBEDTLS_TEST_HOOKS, also try to - * call \link mbedtls_test_hook_error_add \endlink. - * - * \param high high-level error code. See error.h for more details. - * \param low low-level error code. See error.h for more details. - * \param file file where this error code addition occurred. - * \param line line where this error code addition occurred. - */ -static inline int mbedtls_error_add( int high, int low, - const char *file, int line ) -{ -#if defined(MBEDTLS_TEST_HOOKS) - if( *mbedtls_test_hook_error_add != NULL ) - ( *mbedtls_test_hook_error_add )( high, low, file, line ); -#endif - (void)file; - (void)line; - - return( high + low ); -} - -/** - * \brief Translate a mbed TLS error code into a string representation, - * Result is truncated if necessary and always includes a terminating - * null byte. - * - * \param errnum error code - * \param buffer buffer to place representation in - * \param buflen length of the buffer - */ -void mbedtls_strerror( int errnum, char *buffer, size_t buflen ); - -/** - * \brief Translate the high-level part of an Mbed TLS error code into a string - * representation. - * - * This function returns a const pointer to an un-modifiable string. The caller - * must not try to modify the string. It is intended to be used mostly for - * logging purposes. - * - * \param error_code error code - * - * \return The string representation of the error code, or \c NULL if the error - * code is unknown. - */ -const char * mbedtls_high_level_strerr( int error_code ); - -/** - * \brief Translate the low-level part of an Mbed TLS error code into a string - * representation. - * - * This function returns a const pointer to an un-modifiable string. The caller - * must not try to modify the string. It is intended to be used mostly for - * logging purposes. - * - * \param error_code error code - * - * \return The string representation of the error code, or \c NULL if the error - * code is unknown. - */ -const char * mbedtls_low_level_strerr( int error_code ); - -#ifdef __cplusplus -} -#endif - -#endif /* error.h */ diff --git a/windows/mbedtls/gcm.h b/windows/mbedtls/gcm.h deleted file mode 100644 index d1219bb1..00000000 --- a/windows/mbedtls/gcm.h +++ /dev/null @@ -1,379 +0,0 @@ -/** - * \file gcm.h - * - * \brief This file contains GCM definitions and functions. - * - * The Galois/Counter Mode (GCM) for 128-bit block ciphers is defined - * in D. McGrew, J. Viega, The Galois/Counter Mode of Operation - * (GCM), Natl. Inst. Stand. Technol. - * - * For more information on GCM, see NIST SP 800-38D: Recommendation for - * Block Cipher Modes of Operation: Galois/Counter Mode (GCM) and GMAC. - * - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_GCM_H -#define MBEDTLS_GCM_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/cipher.h" - -#include - -#define MBEDTLS_GCM_ENCRYPT 1 -#define MBEDTLS_GCM_DECRYPT 0 - -#define MBEDTLS_ERR_GCM_AUTH_FAILED -0x0012 /**< Authenticated decryption failed. */ -#define MBEDTLS_ERR_GCM_BAD_INPUT -0x0014 /**< Bad input parameters to function. */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_GCM_ALT) - -/** - * \brief The GCM context structure. - */ -typedef struct mbedtls_gcm_context -{ - mbedtls_cipher_context_t MBEDTLS_PRIVATE(cipher_ctx); /*!< The cipher context used. */ - uint64_t MBEDTLS_PRIVATE(HL)[16]; /*!< Precalculated HTable low. */ - uint64_t MBEDTLS_PRIVATE(HH)[16]; /*!< Precalculated HTable high. */ - uint64_t MBEDTLS_PRIVATE(len); /*!< The total length of the encrypted data. */ - uint64_t MBEDTLS_PRIVATE(add_len); /*!< The total length of the additional data. */ - unsigned char MBEDTLS_PRIVATE(base_ectr)[16]; /*!< The first ECTR for tag. */ - unsigned char MBEDTLS_PRIVATE(y)[16]; /*!< The Y working value. */ - unsigned char MBEDTLS_PRIVATE(buf)[16]; /*!< The buf working value. */ - int MBEDTLS_PRIVATE(mode); /*!< The operation to perform: - #MBEDTLS_GCM_ENCRYPT or - #MBEDTLS_GCM_DECRYPT. */ -} -mbedtls_gcm_context; - -#else /* !MBEDTLS_GCM_ALT */ -#include "gcm_alt.h" -#endif /* !MBEDTLS_GCM_ALT */ - -/** - * \brief This function initializes the specified GCM context, - * to make references valid, and prepares the context - * for mbedtls_gcm_setkey() or mbedtls_gcm_free(). - * - * The function does not bind the GCM context to a particular - * cipher, nor set the key. For this purpose, use - * mbedtls_gcm_setkey(). - * - * \param ctx The GCM context to initialize. This must not be \c NULL. - */ -void mbedtls_gcm_init( mbedtls_gcm_context *ctx ); - -/** - * \brief This function associates a GCM context with a - * cipher algorithm and a key. - * - * \param ctx The GCM context. This must be initialized. - * \param cipher The 128-bit block cipher to use. - * \param key The encryption key. This must be a readable buffer of at - * least \p keybits bits. - * \param keybits The key size in bits. Valid options are: - *
  • 128 bits
  • - *
  • 192 bits
  • - *
  • 256 bits
- * - * \return \c 0 on success. - * \return A cipher-specific error code on failure. - */ -int mbedtls_gcm_setkey( mbedtls_gcm_context *ctx, - mbedtls_cipher_id_t cipher, - const unsigned char *key, - unsigned int keybits ); - -/** - * \brief This function performs GCM encryption or decryption of a buffer. - * - * \note For encryption, the output buffer can be the same as the - * input buffer. For decryption, the output buffer cannot be - * the same as input buffer. If the buffers overlap, the output - * buffer must trail at least 8 Bytes behind the input buffer. - * - * \warning When this function performs a decryption, it outputs the - * authentication tag and does not verify that the data is - * authentic. You should use this function to perform encryption - * only. For decryption, use mbedtls_gcm_auth_decrypt() instead. - * - * \param ctx The GCM context to use for encryption or decryption. This - * must be initialized. - * \param mode The operation to perform: - * - #MBEDTLS_GCM_ENCRYPT to perform authenticated encryption. - * The ciphertext is written to \p output and the - * authentication tag is written to \p tag. - * - #MBEDTLS_GCM_DECRYPT to perform decryption. - * The plaintext is written to \p output and the - * authentication tag is written to \p tag. - * Note that this mode is not recommended, because it does - * not verify the authenticity of the data. For this reason, - * you should use mbedtls_gcm_auth_decrypt() instead of - * calling this function in decryption mode. - * \param length The length of the input data, which is equal to the length - * of the output data. - * \param iv The initialization vector. This must be a readable buffer of - * at least \p iv_len Bytes. - * \param iv_len The length of the IV. - * \param add The buffer holding the additional data. This must be of at - * least that size in Bytes. - * \param add_len The length of the additional data. - * \param input The buffer holding the input data. If \p length is greater - * than zero, this must be a readable buffer of at least that - * size in Bytes. - * \param output The buffer for holding the output data. If \p length is greater - * than zero, this must be a writable buffer of at least that - * size in Bytes. - * \param tag_len The length of the tag to generate. - * \param tag The buffer for holding the tag. This must be a writable - * buffer of at least \p tag_len Bytes. - * - * \return \c 0 if the encryption or decryption was performed - * successfully. Note that in #MBEDTLS_GCM_DECRYPT mode, - * this does not indicate that the data is authentic. - * \return #MBEDTLS_ERR_GCM_BAD_INPUT if the lengths or pointers are - * not valid or a cipher-specific error code if the encryption - * or decryption failed. - */ -int mbedtls_gcm_crypt_and_tag( mbedtls_gcm_context *ctx, - int mode, - size_t length, - const unsigned char *iv, - size_t iv_len, - const unsigned char *add, - size_t add_len, - const unsigned char *input, - unsigned char *output, - size_t tag_len, - unsigned char *tag ); - -/** - * \brief This function performs a GCM authenticated decryption of a - * buffer. - * - * \note For decryption, the output buffer cannot be the same as - * input buffer. If the buffers overlap, the output buffer - * must trail at least 8 Bytes behind the input buffer. - * - * \param ctx The GCM context. This must be initialized. - * \param length The length of the ciphertext to decrypt, which is also - * the length of the decrypted plaintext. - * \param iv The initialization vector. This must be a readable buffer - * of at least \p iv_len Bytes. - * \param iv_len The length of the IV. - * \param add The buffer holding the additional data. This must be of at - * least that size in Bytes. - * \param add_len The length of the additional data. - * \param tag The buffer holding the tag to verify. This must be a - * readable buffer of at least \p tag_len Bytes. - * \param tag_len The length of the tag to verify. - * \param input The buffer holding the ciphertext. If \p length is greater - * than zero, this must be a readable buffer of at least that - * size. - * \param output The buffer for holding the decrypted plaintext. If \p length - * is greater than zero, this must be a writable buffer of at - * least that size. - * - * \return \c 0 if successful and authenticated. - * \return #MBEDTLS_ERR_GCM_AUTH_FAILED if the tag does not match. - * \return #MBEDTLS_ERR_GCM_BAD_INPUT if the lengths or pointers are - * not valid or a cipher-specific error code if the decryption - * failed. - */ -int mbedtls_gcm_auth_decrypt( mbedtls_gcm_context *ctx, - size_t length, - const unsigned char *iv, - size_t iv_len, - const unsigned char *add, - size_t add_len, - const unsigned char *tag, - size_t tag_len, - const unsigned char *input, - unsigned char *output ); - -/** - * \brief This function starts a GCM encryption or decryption - * operation. - * - * \param ctx The GCM context. This must be initialized. - * \param mode The operation to perform: #MBEDTLS_GCM_ENCRYPT or - * #MBEDTLS_GCM_DECRYPT. - * \param iv The initialization vector. This must be a readable buffer of - * at least \p iv_len Bytes. - * \param iv_len The length of the IV. - * - * \return \c 0 on success. - */ -int mbedtls_gcm_starts( mbedtls_gcm_context *ctx, - int mode, - const unsigned char *iv, - size_t iv_len ); - -/** - * \brief This function feeds an input buffer as associated data - * (authenticated but not encrypted data) in a GCM - * encryption or decryption operation. - * - * Call this function after mbedtls_gcm_starts() to pass - * the associated data. If the associated data is empty, - * you do not need to call this function. You may not - * call this function after calling mbedtls_cipher_update(). - * - * \param ctx The GCM context. This must have been started with - * mbedtls_gcm_starts() and must not have yet received - * any input with mbedtls_gcm_update(). - * \param add The buffer holding the additional data, or \c NULL - * if \p add_len is \c 0. - * \param add_len The length of the additional data. If \c 0, - * \p add may be \c NULL. - * - * \return \c 0 on success. - */ -int mbedtls_gcm_update_ad( mbedtls_gcm_context *ctx, - const unsigned char *add, - size_t add_len ); - -/** - * \brief This function feeds an input buffer into an ongoing GCM - * encryption or decryption operation. - * - * You may call this function zero, one or more times - * to pass successive parts of the input: the plaintext to - * encrypt, or the ciphertext (not including the tag) to - * decrypt. After the last part of the input, call - * mbedtls_gcm_finish(). - * - * This function may produce output in one of the following - * ways: - * - Immediate output: the output length is always equal - * to the input length. - * - Buffered output: the output consists of a whole number - * of 16-byte blocks. If the total input length so far - * (not including associated data) is 16 \* *B* + *A* - * with *A* < 16 then the total output length is 16 \* *B*. - * - * In particular: - * - It is always correct to call this function with - * \p output_size >= \p input_length + 15. - * - If \p input_length is a multiple of 16 for all the calls - * to this function during an operation, then it is - * correct to use \p output_size = \p input_length. - * - * \note For decryption, the output buffer cannot be the same as - * input buffer. If the buffers overlap, the output buffer - * must trail at least 8 Bytes behind the input buffer. - * - * \param ctx The GCM context. This must be initialized. - * \param input The buffer holding the input data. If \p input_length - * is greater than zero, this must be a readable buffer - * of at least \p input_length bytes. - * \param input_length The length of the input data in bytes. - * \param output The buffer for the output data. If \p output_size - * is greater than zero, this must be a writable buffer of - * of at least \p output_size bytes. - * \param output_size The size of the output buffer in bytes. - * See the function description regarding the output size. - * \param output_length On success, \p *output_length contains the actual - * length of the output written in \p output. - * On failure, the content of \p *output_length is - * unspecified. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_GCM_BAD_INPUT on failure: - * total input length too long, - * unsupported input/output buffer overlap detected, - * or \p output_size too small. - */ -int mbedtls_gcm_update( mbedtls_gcm_context *ctx, - const unsigned char *input, size_t input_length, - unsigned char *output, size_t output_size, - size_t *output_length ); - -/** - * \brief This function finishes the GCM operation and generates - * the authentication tag. - * - * It wraps up the GCM stream, and generates the - * tag. The tag can have a maximum length of 16 Bytes. - * - * \param ctx The GCM context. This must be initialized. - * \param tag The buffer for holding the tag. This must be a writable - * buffer of at least \p tag_len Bytes. - * \param tag_len The length of the tag to generate. This must be at least - * four. - * \param output The buffer for the final output. - * If \p output_size is nonzero, this must be a writable - * buffer of at least \p output_size bytes. - * \param output_size The size of the \p output buffer in bytes. - * This must be large enough for the output that - * mbedtls_gcm_update() has not produced. In particular: - * - If mbedtls_gcm_update() produces immediate output, - * or if the total input size is a multiple of \c 16, - * then mbedtls_gcm_finish() never produces any output, - * so \p output_size can be \c 0. - * - \p output_size never needs to be more than \c 15. - * \param output_length On success, \p *output_length contains the actual - * length of the output written in \p output. - * On failure, the content of \p *output_length is - * unspecified. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_GCM_BAD_INPUT on failure: - * invalid value of \p tag_len, - * or \p output_size too small. - */ -int mbedtls_gcm_finish( mbedtls_gcm_context *ctx, - unsigned char *output, size_t output_size, - size_t *output_length, - unsigned char *tag, size_t tag_len ); - -/** - * \brief This function clears a GCM context and the underlying - * cipher sub-context. - * - * \param ctx The GCM context to clear. If this is \c NULL, the call has - * no effect. Otherwise, this must be initialized. - */ -void mbedtls_gcm_free( mbedtls_gcm_context *ctx ); - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief The GCM checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_gcm_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - - -#endif /* gcm.h */ diff --git a/windows/mbedtls/hkdf.h b/windows/mbedtls/hkdf.h deleted file mode 100644 index 81a62701..00000000 --- a/windows/mbedtls/hkdf.h +++ /dev/null @@ -1,135 +0,0 @@ -/** - * \file hkdf.h - * - * \brief This file contains the HKDF interface. - * - * The HMAC-based Extract-and-Expand Key Derivation Function (HKDF) is - * specified by RFC 5869. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_HKDF_H -#define MBEDTLS_HKDF_H - -#include "mbedtls/build_info.h" - -#include "mbedtls/md.h" - -/** - * \name HKDF Error codes - * \{ - */ -#define MBEDTLS_ERR_HKDF_BAD_INPUT_DATA -0x5F80 /**< Bad input parameters to function. */ -/* \} name */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief This is the HMAC-based Extract-and-Expand Key Derivation Function - * (HKDF). - * - * \param md A hash function; md.size denotes the length of the hash - * function output in bytes. - * \param salt An optional salt value (a non-secret random value); - * if the salt is not provided, a string of all zeros of - * md.size length is used as the salt. - * \param salt_len The length in bytes of the optional \p salt. - * \param ikm The input keying material. - * \param ikm_len The length in bytes of \p ikm. - * \param info An optional context and application specific information - * string. This can be a zero-length string. - * \param info_len The length of \p info in bytes. - * \param okm The output keying material of \p okm_len bytes. - * \param okm_len The length of the output keying material in bytes. This - * must be less than or equal to 255 * md.size bytes. - * - * \return 0 on success. - * \return #MBEDTLS_ERR_HKDF_BAD_INPUT_DATA when the parameters are invalid. - * \return An MBEDTLS_ERR_MD_* error for errors returned from the underlying - * MD layer. - */ -int mbedtls_hkdf( const mbedtls_md_info_t *md, const unsigned char *salt, - size_t salt_len, const unsigned char *ikm, size_t ikm_len, - const unsigned char *info, size_t info_len, - unsigned char *okm, size_t okm_len ); - -/** - * \brief Take the input keying material \p ikm and extract from it a - * fixed-length pseudorandom key \p prk. - * - * \warning This function should only be used if the security of it has been - * studied and established in that particular context (eg. TLS 1.3 - * key schedule). For standard HKDF security guarantees use - * \c mbedtls_hkdf instead. - * - * \param md A hash function; md.size denotes the length of the - * hash function output in bytes. - * \param salt An optional salt value (a non-secret random value); - * if the salt is not provided, a string of all zeros - * of md.size length is used as the salt. - * \param salt_len The length in bytes of the optional \p salt. - * \param ikm The input keying material. - * \param ikm_len The length in bytes of \p ikm. - * \param[out] prk A pseudorandom key of at least md.size bytes. - * - * \return 0 on success. - * \return #MBEDTLS_ERR_HKDF_BAD_INPUT_DATA when the parameters are invalid. - * \return An MBEDTLS_ERR_MD_* error for errors returned from the underlying - * MD layer. - */ -int mbedtls_hkdf_extract( const mbedtls_md_info_t *md, - const unsigned char *salt, size_t salt_len, - const unsigned char *ikm, size_t ikm_len, - unsigned char *prk ); - -/** - * \brief Expand the supplied \p prk into several additional pseudorandom - * keys, which is the output of the HKDF. - * - * \warning This function should only be used if the security of it has been - * studied and established in that particular context (eg. TLS 1.3 - * key schedule). For standard HKDF security guarantees use - * \c mbedtls_hkdf instead. - * - * \param md A hash function; md.size denotes the length of the hash - * function output in bytes. - * \param prk A pseudorandom key of at least md.size bytes. \p prk is - * usually the output from the HKDF extract step. - * \param prk_len The length in bytes of \p prk. - * \param info An optional context and application specific information - * string. This can be a zero-length string. - * \param info_len The length of \p info in bytes. - * \param okm The output keying material of \p okm_len bytes. - * \param okm_len The length of the output keying material in bytes. This - * must be less than or equal to 255 * md.size bytes. - * - * \return 0 on success. - * \return #MBEDTLS_ERR_HKDF_BAD_INPUT_DATA when the parameters are invalid. - * \return An MBEDTLS_ERR_MD_* error for errors returned from the underlying - * MD layer. - */ -int mbedtls_hkdf_expand( const mbedtls_md_info_t *md, const unsigned char *prk, - size_t prk_len, const unsigned char *info, - size_t info_len, unsigned char *okm, size_t okm_len ); - -#ifdef __cplusplus -} -#endif - -#endif /* hkdf.h */ diff --git a/windows/mbedtls/hmac_drbg.h b/windows/mbedtls/hmac_drbg.h deleted file mode 100644 index 0df8ee85..00000000 --- a/windows/mbedtls/hmac_drbg.h +++ /dev/null @@ -1,443 +0,0 @@ -/** - * \file hmac_drbg.h - * - * \brief The HMAC_DRBG pseudorandom generator. - * - * This module implements the HMAC_DRBG pseudorandom generator described - * in NIST SP 800-90A: Recommendation for Random Number Generation Using - * Deterministic Random Bit Generators. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_HMAC_DRBG_H -#define MBEDTLS_HMAC_DRBG_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/md.h" - -#if defined(MBEDTLS_THREADING_C) -#include "mbedtls/threading.h" -#endif - -/* - * Error codes - */ -#define MBEDTLS_ERR_HMAC_DRBG_REQUEST_TOO_BIG -0x0003 /**< Too many random requested in single call. */ -#define MBEDTLS_ERR_HMAC_DRBG_INPUT_TOO_BIG -0x0005 /**< Input too large (Entropy + additional). */ -#define MBEDTLS_ERR_HMAC_DRBG_FILE_IO_ERROR -0x0007 /**< Read/write error in file. */ -#define MBEDTLS_ERR_HMAC_DRBG_ENTROPY_SOURCE_FAILED -0x0009 /**< The entropy source failed. */ - -/** - * \name SECTION: Module settings - * - * The configuration options you can set for this module are in this section. - * Either change them in mbedtls_config.h or define them on the compiler command line. - * \{ - */ - -#if !defined(MBEDTLS_HMAC_DRBG_RESEED_INTERVAL) -#define MBEDTLS_HMAC_DRBG_RESEED_INTERVAL 10000 /**< Interval before reseed is performed by default */ -#endif - -#if !defined(MBEDTLS_HMAC_DRBG_MAX_INPUT) -#define MBEDTLS_HMAC_DRBG_MAX_INPUT 256 /**< Maximum number of additional input bytes */ -#endif - -#if !defined(MBEDTLS_HMAC_DRBG_MAX_REQUEST) -#define MBEDTLS_HMAC_DRBG_MAX_REQUEST 1024 /**< Maximum number of requested bytes per call */ -#endif - -#if !defined(MBEDTLS_HMAC_DRBG_MAX_SEED_INPUT) -#define MBEDTLS_HMAC_DRBG_MAX_SEED_INPUT 384 /**< Maximum size of (re)seed buffer */ -#endif - -/* \} name SECTION: Module settings */ - -#define MBEDTLS_HMAC_DRBG_PR_OFF 0 /**< No prediction resistance */ -#define MBEDTLS_HMAC_DRBG_PR_ON 1 /**< Prediction resistance enabled */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * HMAC_DRBG context. - */ -typedef struct mbedtls_hmac_drbg_context -{ - /* Working state: the key K is not stored explicitly, - * but is implied by the HMAC context */ - mbedtls_md_context_t MBEDTLS_PRIVATE(md_ctx); /*!< HMAC context (inc. K) */ - unsigned char MBEDTLS_PRIVATE(V)[MBEDTLS_MD_MAX_SIZE]; /*!< V in the spec */ - int MBEDTLS_PRIVATE(reseed_counter); /*!< reseed counter */ - - /* Administrative state */ - size_t MBEDTLS_PRIVATE(entropy_len); /*!< entropy bytes grabbed on each (re)seed */ - int MBEDTLS_PRIVATE(prediction_resistance); /*!< enable prediction resistance (Automatic - reseed before every random generation) */ - int MBEDTLS_PRIVATE(reseed_interval); /*!< reseed interval */ - - /* Callbacks */ - int (*MBEDTLS_PRIVATE(f_entropy))(void *, unsigned char *, size_t); /*!< entropy function */ - void *MBEDTLS_PRIVATE(p_entropy); /*!< context for the entropy function */ - -#if defined(MBEDTLS_THREADING_C) - /* Invariant: the mutex is initialized if and only if - * md_ctx->md_info != NULL. This means that the mutex is initialized - * during the initial seeding in mbedtls_hmac_drbg_seed() or - * mbedtls_hmac_drbg_seed_buf() and freed in mbedtls_ctr_drbg_free(). - * - * Note that this invariant may change without notice. Do not rely on it - * and do not access the mutex directly in application code. - */ - mbedtls_threading_mutex_t MBEDTLS_PRIVATE(mutex); -#endif -} mbedtls_hmac_drbg_context; - -/** - * \brief HMAC_DRBG context initialization. - * - * This function makes the context ready for mbedtls_hmac_drbg_seed(), - * mbedtls_hmac_drbg_seed_buf() or mbedtls_hmac_drbg_free(). - * - * \note The reseed interval is #MBEDTLS_HMAC_DRBG_RESEED_INTERVAL - * by default. Override this value by calling - * mbedtls_hmac_drbg_set_reseed_interval(). - * - * \param ctx HMAC_DRBG context to be initialized. - */ -void mbedtls_hmac_drbg_init( mbedtls_hmac_drbg_context *ctx ); - -/** - * \brief HMAC_DRBG initial seeding. - * - * Set the initial seed and set up the entropy source for future reseeds. - * - * A typical choice for the \p f_entropy and \p p_entropy parameters is - * to use the entropy module: - * - \p f_entropy is mbedtls_entropy_func(); - * - \p p_entropy is an instance of ::mbedtls_entropy_context initialized - * with mbedtls_entropy_init() (which registers the platform's default - * entropy sources). - * - * You can provide a personalization string in addition to the - * entropy source, to make this instantiation as unique as possible. - * - * \note By default, the security strength as defined by NIST is: - * - 128 bits if \p md_info is SHA-1; - * - 192 bits if \p md_info is SHA-224; - * - 256 bits if \p md_info is SHA-256, SHA-384 or SHA-512. - * Note that SHA-256 is just as efficient as SHA-224. - * The security strength can be reduced if a smaller - * entropy length is set with - * mbedtls_hmac_drbg_set_entropy_len(). - * - * \note The default entropy length is the security strength - * (converted from bits to bytes). You can override - * it by calling mbedtls_hmac_drbg_set_entropy_len(). - * - * \note During the initial seeding, this function calls - * the entropy source to obtain a nonce - * whose length is half the entropy length. - */ -#if defined(MBEDTLS_THREADING_C) -/** - * \note When Mbed TLS is built with threading support, - * after this function returns successfully, - * it is safe to call mbedtls_hmac_drbg_random() - * from multiple threads. Other operations, including - * reseeding, are not thread-safe. - */ -#endif /* MBEDTLS_THREADING_C */ -/** - * \param ctx HMAC_DRBG context to be seeded. - * \param md_info MD algorithm to use for HMAC_DRBG. - * \param f_entropy The entropy callback, taking as arguments the - * \p p_entropy context, the buffer to fill, and the - * length of the buffer. - * \p f_entropy is always called with a length that is - * less than or equal to the entropy length. - * \param p_entropy The entropy context to pass to \p f_entropy. - * \param custom The personalization string. - * This can be \c NULL, in which case the personalization - * string is empty regardless of the value of \p len. - * \param len The length of the personalization string. - * This must be at most #MBEDTLS_HMAC_DRBG_MAX_INPUT - * and also at most - * #MBEDTLS_HMAC_DRBG_MAX_SEED_INPUT - \p entropy_len * 3 / 2 - * where \p entropy_len is the entropy length - * described above. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA if \p md_info is - * invalid. - * \return #MBEDTLS_ERR_MD_ALLOC_FAILED if there was not enough - * memory to allocate context data. - * \return #MBEDTLS_ERR_HMAC_DRBG_ENTROPY_SOURCE_FAILED - * if the call to \p f_entropy failed. - */ -int mbedtls_hmac_drbg_seed( mbedtls_hmac_drbg_context *ctx, - const mbedtls_md_info_t * md_info, - int (*f_entropy)(void *, unsigned char *, size_t), - void *p_entropy, - const unsigned char *custom, - size_t len ); - -/** - * \brief Initilisation of simpified HMAC_DRBG (never reseeds). - * - * This function is meant for use in algorithms that need a pseudorandom - * input such as deterministic ECDSA. - */ -#if defined(MBEDTLS_THREADING_C) -/** - * \note When Mbed TLS is built with threading support, - * after this function returns successfully, - * it is safe to call mbedtls_hmac_drbg_random() - * from multiple threads. Other operations, including - * reseeding, are not thread-safe. - */ -#endif /* MBEDTLS_THREADING_C */ -/** - * \param ctx HMAC_DRBG context to be initialised. - * \param md_info MD algorithm to use for HMAC_DRBG. - * \param data Concatenation of the initial entropy string and - * the additional data. - * \param data_len Length of \p data in bytes. - * - * \return \c 0 if successful. or - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA if \p md_info is - * invalid. - * \return #MBEDTLS_ERR_MD_ALLOC_FAILED if there was not enough - * memory to allocate context data. - */ -int mbedtls_hmac_drbg_seed_buf( mbedtls_hmac_drbg_context *ctx, - const mbedtls_md_info_t * md_info, - const unsigned char *data, size_t data_len ); - -/** - * \brief This function turns prediction resistance on or off. - * The default value is off. - * - * \note If enabled, entropy is gathered at the beginning of - * every call to mbedtls_hmac_drbg_random_with_add() - * or mbedtls_hmac_drbg_random(). - * Only use this if your entropy source has sufficient - * throughput. - * - * \param ctx The HMAC_DRBG context. - * \param resistance #MBEDTLS_HMAC_DRBG_PR_ON or #MBEDTLS_HMAC_DRBG_PR_OFF. - */ -void mbedtls_hmac_drbg_set_prediction_resistance( mbedtls_hmac_drbg_context *ctx, - int resistance ); - -/** - * \brief This function sets the amount of entropy grabbed on each - * seed or reseed. - * - * See the documentation of mbedtls_hmac_drbg_seed() for the default value. - * - * \param ctx The HMAC_DRBG context. - * \param len The amount of entropy to grab, in bytes. - */ -void mbedtls_hmac_drbg_set_entropy_len( mbedtls_hmac_drbg_context *ctx, - size_t len ); - -/** - * \brief Set the reseed interval. - * - * The reseed interval is the number of calls to mbedtls_hmac_drbg_random() - * or mbedtls_hmac_drbg_random_with_add() after which the entropy function - * is called again. - * - * The default value is #MBEDTLS_HMAC_DRBG_RESEED_INTERVAL. - * - * \param ctx The HMAC_DRBG context. - * \param interval The reseed interval. - */ -void mbedtls_hmac_drbg_set_reseed_interval( mbedtls_hmac_drbg_context *ctx, - int interval ); - -/** - * \brief This function updates the state of the HMAC_DRBG context. - * - * \note This function is not thread-safe. It is not safe - * to call this function if another thread might be - * concurrently obtaining random numbers from the same - * context or updating or reseeding the same context. - * - * \param ctx The HMAC_DRBG context. - * \param additional The data to update the state with. - * If this is \c NULL, there is no additional data. - * \param add_len Length of \p additional in bytes. - * Unused if \p additional is \c NULL. - * - * \return \c 0 on success, or an error from the underlying - * hash calculation. - */ -int mbedtls_hmac_drbg_update( mbedtls_hmac_drbg_context *ctx, - const unsigned char *additional, size_t add_len ); - -/** - * \brief This function reseeds the HMAC_DRBG context, that is - * extracts data from the entropy source. - * - * \note This function is not thread-safe. It is not safe - * to call this function if another thread might be - * concurrently obtaining random numbers from the same - * context or updating or reseeding the same context. - * - * \param ctx The HMAC_DRBG context. - * \param additional Additional data to add to the state. - * If this is \c NULL, there is no additional data - * and \p len should be \c 0. - * \param len The length of the additional data. - * This must be at most #MBEDTLS_HMAC_DRBG_MAX_INPUT - * and also at most - * #MBEDTLS_HMAC_DRBG_MAX_SEED_INPUT - \p entropy_len - * where \p entropy_len is the entropy length - * (see mbedtls_hmac_drbg_set_entropy_len()). - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_HMAC_DRBG_ENTROPY_SOURCE_FAILED - * if a call to the entropy function failed. - */ -int mbedtls_hmac_drbg_reseed( mbedtls_hmac_drbg_context *ctx, - const unsigned char *additional, size_t len ); - -/** - * \brief This function updates an HMAC_DRBG instance with additional - * data and uses it to generate random data. - * - * This function automatically reseeds if the reseed counter is exceeded - * or prediction resistance is enabled. - * - * \note This function is not thread-safe. It is not safe - * to call this function if another thread might be - * concurrently obtaining random numbers from the same - * context or updating or reseeding the same context. - * - * \param p_rng The HMAC_DRBG context. This must be a pointer to a - * #mbedtls_hmac_drbg_context structure. - * \param output The buffer to fill. - * \param output_len The length of the buffer in bytes. - * This must be at most #MBEDTLS_HMAC_DRBG_MAX_REQUEST. - * \param additional Additional data to update with. - * If this is \c NULL, there is no additional data - * and \p add_len should be \c 0. - * \param add_len The length of the additional data. - * This must be at most #MBEDTLS_HMAC_DRBG_MAX_INPUT. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_HMAC_DRBG_ENTROPY_SOURCE_FAILED - * if a call to the entropy source failed. - * \return #MBEDTLS_ERR_HMAC_DRBG_REQUEST_TOO_BIG if - * \p output_len > #MBEDTLS_HMAC_DRBG_MAX_REQUEST. - * \return #MBEDTLS_ERR_HMAC_DRBG_INPUT_TOO_BIG if - * \p add_len > #MBEDTLS_HMAC_DRBG_MAX_INPUT. - */ -int mbedtls_hmac_drbg_random_with_add( void *p_rng, - unsigned char *output, size_t output_len, - const unsigned char *additional, - size_t add_len ); - -/** - * \brief This function uses HMAC_DRBG to generate random data. - * - * This function automatically reseeds if the reseed counter is exceeded - * or prediction resistance is enabled. - */ -#if defined(MBEDTLS_THREADING_C) -/** - * \note When Mbed TLS is built with threading support, - * it is safe to call mbedtls_ctr_drbg_random() - * from multiple threads. Other operations, including - * reseeding, are not thread-safe. - */ -#endif /* MBEDTLS_THREADING_C */ -/** - * \param p_rng The HMAC_DRBG context. This must be a pointer to a - * #mbedtls_hmac_drbg_context structure. - * \param output The buffer to fill. - * \param out_len The length of the buffer in bytes. - * This must be at most #MBEDTLS_HMAC_DRBG_MAX_REQUEST. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_HMAC_DRBG_ENTROPY_SOURCE_FAILED - * if a call to the entropy source failed. - * \return #MBEDTLS_ERR_HMAC_DRBG_REQUEST_TOO_BIG if - * \p out_len > #MBEDTLS_HMAC_DRBG_MAX_REQUEST. - */ -int mbedtls_hmac_drbg_random( void *p_rng, unsigned char *output, size_t out_len ); - -/** - * \brief This function resets HMAC_DRBG context to the state immediately - * after initial call of mbedtls_hmac_drbg_init(). - * - * \param ctx The HMAC_DRBG context to free. - */ -void mbedtls_hmac_drbg_free( mbedtls_hmac_drbg_context *ctx ); - -#if defined(MBEDTLS_FS_IO) -/** - * \brief This function writes a seed file. - * - * \param ctx The HMAC_DRBG context. - * \param path The name of the file. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_HMAC_DRBG_FILE_IO_ERROR on file error. - * \return #MBEDTLS_ERR_HMAC_DRBG_ENTROPY_SOURCE_FAILED on reseed - * failure. - */ -int mbedtls_hmac_drbg_write_seed_file( mbedtls_hmac_drbg_context *ctx, const char *path ); - -/** - * \brief This function reads and updates a seed file. The seed - * is added to this instance. - * - * \param ctx The HMAC_DRBG context. - * \param path The name of the file. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_HMAC_DRBG_FILE_IO_ERROR on file error. - * \return #MBEDTLS_ERR_HMAC_DRBG_ENTROPY_SOURCE_FAILED on - * reseed failure. - * \return #MBEDTLS_ERR_HMAC_DRBG_INPUT_TOO_BIG if the existing - * seed file is too large. - */ -int mbedtls_hmac_drbg_update_seed_file( mbedtls_hmac_drbg_context *ctx, const char *path ); -#endif /* MBEDTLS_FS_IO */ - - -#if defined(MBEDTLS_SELF_TEST) -/** - * \brief The HMAC_DRBG Checkup routine. - * - * \return \c 0 if successful. - * \return \c 1 if the test failed. - */ -int mbedtls_hmac_drbg_self_test( int verbose ); -#endif - -#ifdef __cplusplus -} -#endif - -#endif /* hmac_drbg.h */ diff --git a/windows/mbedtls/mbedtls_config.h b/windows/mbedtls/mbedtls_config.h deleted file mode 100644 index a60db7e9..00000000 --- a/windows/mbedtls/mbedtls_config.h +++ /dev/null @@ -1,3263 +0,0 @@ -/** - * \file mbedtls_config.h - * - * \brief Configuration options (set of defines) - * - * This set of compile-time options may be used to enable - * or disable features selectively, and reduce the global - * memory footprint. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -/** - * This is an optional version symbol that enables comatibility handling of - * config files. - * - * It is equal to the #MBEDTLS_VERSION_NUMBER of the Mbed TLS version that - * introduced the config format we want to be compatible with. - */ -//#define MBEDTLS_CONFIG_VERSION 0x03000000 - -/** - * \name SECTION: System support - * - * This section sets system specific settings. - * \{ - */ - -/** - * \def MBEDTLS_HAVE_ASM - * - * The compiler has support for asm(). - * - * Requires support for asm() in compiler. - * - * Used in: - * library/aria.c - * library/bn_mul.h - * - * Required by: - * MBEDTLS_AESNI_C - * MBEDTLS_PADLOCK_C - * - * Comment to disable the use of assembly code. - */ -#define MBEDTLS_HAVE_ASM - -/** - * \def MBEDTLS_NO_UDBL_DIVISION - * - * The platform lacks support for double-width integer division (64-bit - * division on a 32-bit platform, 128-bit division on a 64-bit platform). - * - * Used in: - * include/mbedtls/bignum.h - * library/bignum.c - * - * The bignum code uses double-width division to speed up some operations. - * Double-width division is often implemented in software that needs to - * be linked with the program. The presence of a double-width integer - * type is usually detected automatically through preprocessor macros, - * but the automatic detection cannot know whether the code needs to - * and can be linked with an implementation of division for that type. - * By default division is assumed to be usable if the type is present. - * Uncomment this option to prevent the use of double-width division. - * - * Note that division for the native integer type is always required. - * Furthermore, a 64-bit type is always required even on a 32-bit - * platform, but it need not support multiplication or division. In some - * cases it is also desirable to disable some double-width operations. For - * example, if double-width division is implemented in software, disabling - * it can reduce code size in some embedded targets. - */ -//#define MBEDTLS_NO_UDBL_DIVISION - -/** - * \def MBEDTLS_NO_64BIT_MULTIPLICATION - * - * The platform lacks support for 32x32 -> 64-bit multiplication. - * - * Used in: - * library/poly1305.c - * - * Some parts of the library may use multiplication of two unsigned 32-bit - * operands with a 64-bit result in order to speed up computations. On some - * platforms, this is not available in hardware and has to be implemented in - * software, usually in a library provided by the toolchain. - * - * Sometimes it is not desirable to have to link to that library. This option - * removes the dependency of that library on platforms that lack a hardware - * 64-bit multiplier by embedding a software implementation in Mbed TLS. - * - * Note that depending on the compiler, this may decrease performance compared - * to using the library function provided by the toolchain. - */ -//#define MBEDTLS_NO_64BIT_MULTIPLICATION - -/** - * \def MBEDTLS_HAVE_SSE2 - * - * CPU supports SSE2 instruction set. - * - * Uncomment if the CPU supports SSE2 (IA-32 specific). - */ -//#define MBEDTLS_HAVE_SSE2 - -/** - * \def MBEDTLS_HAVE_TIME - * - * System has time.h and time(). - * The time does not need to be correct, only time differences are used, - * by contrast with MBEDTLS_HAVE_TIME_DATE - * - * Defining MBEDTLS_HAVE_TIME allows you to specify MBEDTLS_PLATFORM_TIME_ALT, - * MBEDTLS_PLATFORM_TIME_MACRO, MBEDTLS_PLATFORM_TIME_TYPE_MACRO and - * MBEDTLS_PLATFORM_STD_TIME. - * - * Comment if your system does not support time functions - */ -#define MBEDTLS_HAVE_TIME - -/** - * \def MBEDTLS_HAVE_TIME_DATE - * - * System has time.h, time(), and an implementation for - * mbedtls_platform_gmtime_r() (see below). - * The time needs to be correct (not necessarily very accurate, but at least - * the date should be correct). This is used to verify the validity period of - * X.509 certificates. - * - * Comment if your system does not have a correct clock. - * - * \note mbedtls_platform_gmtime_r() is an abstraction in platform_util.h that - * behaves similarly to the gmtime_r() function from the C standard. Refer to - * the documentation for mbedtls_platform_gmtime_r() for more information. - * - * \note It is possible to configure an implementation for - * mbedtls_platform_gmtime_r() at compile-time by using the macro - * MBEDTLS_PLATFORM_GMTIME_R_ALT. - */ -#define MBEDTLS_HAVE_TIME_DATE - -/** - * \def MBEDTLS_PLATFORM_MEMORY - * - * Enable the memory allocation layer. - * - * By default mbed TLS uses the system-provided calloc() and free(). - * This allows different allocators (self-implemented or provided) to be - * provided to the platform abstraction layer. - * - * Enabling MBEDTLS_PLATFORM_MEMORY without the - * MBEDTLS_PLATFORM_{FREE,CALLOC}_MACROs will provide - * "mbedtls_platform_set_calloc_free()" allowing you to set an alternative calloc() and - * free() function pointer at runtime. - * - * Enabling MBEDTLS_PLATFORM_MEMORY and specifying - * MBEDTLS_PLATFORM_{CALLOC,FREE}_MACROs will allow you to specify the - * alternate function at compile time. - * - * Requires: MBEDTLS_PLATFORM_C - * - * Enable this layer to allow use of alternative memory allocators. - */ -//#define MBEDTLS_PLATFORM_MEMORY - -/** - * \def MBEDTLS_PLATFORM_NO_STD_FUNCTIONS - * - * Do not assign standard functions in the platform layer (e.g. calloc() to - * MBEDTLS_PLATFORM_STD_CALLOC and printf() to MBEDTLS_PLATFORM_STD_PRINTF) - * - * This makes sure there are no linking errors on platforms that do not support - * these functions. You will HAVE to provide alternatives, either at runtime - * via the platform_set_xxx() functions or at compile time by setting - * the MBEDTLS_PLATFORM_STD_XXX defines, or enabling a - * MBEDTLS_PLATFORM_XXX_MACRO. - * - * Requires: MBEDTLS_PLATFORM_C - * - * Uncomment to prevent default assignment of standard functions in the - * platform layer. - */ -//#define MBEDTLS_PLATFORM_NO_STD_FUNCTIONS - -/** - * \def MBEDTLS_PLATFORM_EXIT_ALT - * - * MBEDTLS_PLATFORM_XXX_ALT: Uncomment a macro to let mbed TLS support the - * function in the platform abstraction layer. - * - * Example: In case you uncomment MBEDTLS_PLATFORM_PRINTF_ALT, mbed TLS will - * provide a function "mbedtls_platform_set_printf()" that allows you to set an - * alternative printf function pointer. - * - * All these define require MBEDTLS_PLATFORM_C to be defined! - * - * \note MBEDTLS_PLATFORM_SNPRINTF_ALT is required on Windows; - * it will be enabled automatically by check_config.h - * - * \warning MBEDTLS_PLATFORM_XXX_ALT cannot be defined at the same time as - * MBEDTLS_PLATFORM_XXX_MACRO! - * - * Requires: MBEDTLS_PLATFORM_TIME_ALT requires MBEDTLS_HAVE_TIME - * - * Uncomment a macro to enable alternate implementation of specific base - * platform function - */ -//#define MBEDTLS_PLATFORM_EXIT_ALT -//#define MBEDTLS_PLATFORM_TIME_ALT -//#define MBEDTLS_PLATFORM_FPRINTF_ALT -//#define MBEDTLS_PLATFORM_PRINTF_ALT -//#define MBEDTLS_PLATFORM_SNPRINTF_ALT -//#define MBEDTLS_PLATFORM_VSNPRINTF_ALT -//#define MBEDTLS_PLATFORM_NV_SEED_ALT -//#define MBEDTLS_PLATFORM_SETUP_TEARDOWN_ALT - -/** - * \def MBEDTLS_DEPRECATED_WARNING - * - * Mark deprecated functions and features so that they generate a warning if - * used. Functionality deprecated in one version will usually be removed in the - * next version. You can enable this to help you prepare the transition to a - * new major version by making sure your code is not using this functionality. - * - * This only works with GCC and Clang. With other compilers, you may want to - * use MBEDTLS_DEPRECATED_REMOVED - * - * Uncomment to get warnings on using deprecated functions and features. - */ -//#define MBEDTLS_DEPRECATED_WARNING - -/** - * \def MBEDTLS_DEPRECATED_REMOVED - * - * Remove deprecated functions and features so that they generate an error if - * used. Functionality deprecated in one version will usually be removed in the - * next version. You can enable this to help you prepare the transition to a - * new major version by making sure your code is not using this functionality. - * - * Uncomment to get errors on using deprecated functions and features. - */ -//#define MBEDTLS_DEPRECATED_REMOVED - -/* \} name SECTION: System support */ - -/** - * \name SECTION: mbed TLS feature support - * - * This section sets support for features that are or are not needed - * within the modules that are enabled. - * \{ - */ - -/** - * \def MBEDTLS_TIMING_ALT - * - * Uncomment to provide your own alternate implementation for - * mbedtls_timing_get_timer(), mbedtls_set_alarm(), mbedtls_set/get_delay() - * - * Only works if you have MBEDTLS_TIMING_C enabled. - * - * You will need to provide a header "timing_alt.h" and an implementation at - * compile time. - */ -//#define MBEDTLS_TIMING_ALT - -/** - * \def MBEDTLS_AES_ALT - * - * MBEDTLS__MODULE_NAME__ALT: Uncomment a macro to let mbed TLS use your - * alternate core implementation of a symmetric crypto, an arithmetic or hash - * module (e.g. platform specific assembly optimized implementations). Keep - * in mind that the function prototypes should remain the same. - * - * This replaces the whole module. If you only want to replace one of the - * functions, use one of the MBEDTLS__FUNCTION_NAME__ALT flags. - * - * Example: In case you uncomment MBEDTLS_AES_ALT, mbed TLS will no longer - * provide the "struct mbedtls_aes_context" definition and omit the base - * function declarations and implementations. "aes_alt.h" will be included from - * "aes.h" to include the new function definitions. - * - * Uncomment a macro to enable alternate implementation of the corresponding - * module. - * - * \warning MD5, DES and SHA-1 are considered weak and their - * use constitutes a security risk. If possible, we recommend - * avoiding dependencies on them, and considering stronger message - * digests and ciphers instead. - * - */ -//#define MBEDTLS_AES_ALT -//#define MBEDTLS_ARIA_ALT -//#define MBEDTLS_CAMELLIA_ALT -//#define MBEDTLS_CCM_ALT -//#define MBEDTLS_CHACHA20_ALT -//#define MBEDTLS_CHACHAPOLY_ALT -//#define MBEDTLS_CMAC_ALT -//#define MBEDTLS_DES_ALT -//#define MBEDTLS_DHM_ALT -//#define MBEDTLS_ECJPAKE_ALT -//#define MBEDTLS_GCM_ALT -//#define MBEDTLS_NIST_KW_ALT -//#define MBEDTLS_MD5_ALT -//#define MBEDTLS_POLY1305_ALT -//#define MBEDTLS_RIPEMD160_ALT -//#define MBEDTLS_RSA_ALT -//#define MBEDTLS_SHA1_ALT -//#define MBEDTLS_SHA256_ALT -//#define MBEDTLS_SHA512_ALT - -/* - * When replacing the elliptic curve module, pleace consider, that it is - * implemented with two .c files: - * - ecp.c - * - ecp_curves.c - * You can replace them very much like all the other MBEDTLS__MODULE_NAME__ALT - * macros as described above. The only difference is that you have to make sure - * that you provide functionality for both .c files. - */ -//#define MBEDTLS_ECP_ALT - -/** - * \def MBEDTLS_SHA256_PROCESS_ALT - * - * MBEDTLS__FUNCTION_NAME__ALT: Uncomment a macro to let mbed TLS use you - * alternate core implementation of symmetric crypto or hash function. Keep in - * mind that function prototypes should remain the same. - * - * This replaces only one function. The header file from mbed TLS is still - * used, in contrast to the MBEDTLS__MODULE_NAME__ALT flags. - * - * Example: In case you uncomment MBEDTLS_SHA256_PROCESS_ALT, mbed TLS will - * no longer provide the mbedtls_sha1_process() function, but it will still provide - * the other function (using your mbedtls_sha1_process() function) and the definition - * of mbedtls_sha1_context, so your implementation of mbedtls_sha1_process must be compatible - * with this definition. - * - * \note If you use the AES_xxx_ALT macros, then it is recommended to also set - * MBEDTLS_AES_ROM_TABLES in order to help the linker garbage-collect the AES - * tables. - * - * Uncomment a macro to enable alternate implementation of the corresponding - * function. - * - * \warning MD5, DES and SHA-1 are considered weak and their use - * constitutes a security risk. If possible, we recommend avoiding - * dependencies on them, and considering stronger message digests - * and ciphers instead. - * - * \warning If both MBEDTLS_ECDSA_SIGN_ALT and MBEDTLS_ECDSA_DETERMINISTIC are - * enabled, then the deterministic ECDH signature functions pass the - * the static HMAC-DRBG as RNG to mbedtls_ecdsa_sign(). Therefore - * alternative implementations should use the RNG only for generating - * the ephemeral key and nothing else. If this is not possible, then - * MBEDTLS_ECDSA_DETERMINISTIC should be disabled and an alternative - * implementation should be provided for mbedtls_ecdsa_sign_det_ext(). - * - */ -//#define MBEDTLS_MD5_PROCESS_ALT -//#define MBEDTLS_RIPEMD160_PROCESS_ALT -//#define MBEDTLS_SHA1_PROCESS_ALT -//#define MBEDTLS_SHA256_PROCESS_ALT -//#define MBEDTLS_SHA512_PROCESS_ALT -//#define MBEDTLS_DES_SETKEY_ALT -//#define MBEDTLS_DES_CRYPT_ECB_ALT -//#define MBEDTLS_DES3_CRYPT_ECB_ALT -//#define MBEDTLS_AES_SETKEY_ENC_ALT -//#define MBEDTLS_AES_SETKEY_DEC_ALT -//#define MBEDTLS_AES_ENCRYPT_ALT -//#define MBEDTLS_AES_DECRYPT_ALT -//#define MBEDTLS_ECDH_GEN_PUBLIC_ALT -//#define MBEDTLS_ECDH_COMPUTE_SHARED_ALT -//#define MBEDTLS_ECDSA_VERIFY_ALT -//#define MBEDTLS_ECDSA_SIGN_ALT -//#define MBEDTLS_ECDSA_GENKEY_ALT - -/** - * \def MBEDTLS_ECP_INTERNAL_ALT - * - * Expose a part of the internal interface of the Elliptic Curve Point module. - * - * MBEDTLS_ECP__FUNCTION_NAME__ALT: Uncomment a macro to let mbed TLS use your - * alternative core implementation of elliptic curve arithmetic. Keep in mind - * that function prototypes should remain the same. - * - * This partially replaces one function. The header file from mbed TLS is still - * used, in contrast to the MBEDTLS_ECP_ALT flag. The original implementation - * is still present and it is used for group structures not supported by the - * alternative. - * - * The original implementation can in addition be removed by setting the - * MBEDTLS_ECP_NO_FALLBACK option, in which case any function for which the - * corresponding MBEDTLS_ECP__FUNCTION_NAME__ALT macro is defined will not be - * able to fallback to curves not supported by the alternative implementation. - * - * Any of these options become available by defining MBEDTLS_ECP_INTERNAL_ALT - * and implementing the following functions: - * unsigned char mbedtls_internal_ecp_grp_capable( - * const mbedtls_ecp_group *grp ) - * int mbedtls_internal_ecp_init( const mbedtls_ecp_group *grp ) - * void mbedtls_internal_ecp_free( const mbedtls_ecp_group *grp ) - * The mbedtls_internal_ecp_grp_capable function should return 1 if the - * replacement functions implement arithmetic for the given group and 0 - * otherwise. - * The functions mbedtls_internal_ecp_init and mbedtls_internal_ecp_free are - * called before and after each point operation and provide an opportunity to - * implement optimized set up and tear down instructions. - * - * Example: In case you set MBEDTLS_ECP_INTERNAL_ALT and - * MBEDTLS_ECP_DOUBLE_JAC_ALT, mbed TLS will still provide the ecp_double_jac() - * function, but will use your mbedtls_internal_ecp_double_jac() if the group - * for the operation is supported by your implementation (i.e. your - * mbedtls_internal_ecp_grp_capable() function returns 1 for this group). If the - * group is not supported by your implementation, then the original mbed TLS - * implementation of ecp_double_jac() is used instead, unless this fallback - * behaviour is disabled by setting MBEDTLS_ECP_NO_FALLBACK (in which case - * ecp_double_jac() will return MBEDTLS_ERR_ECP_FEATURE_UNAVAILABLE). - * - * The function prototypes and the definition of mbedtls_ecp_group and - * mbedtls_ecp_point will not change based on MBEDTLS_ECP_INTERNAL_ALT, so your - * implementation of mbedtls_internal_ecp__function_name__ must be compatible - * with their definitions. - * - * Uncomment a macro to enable alternate implementation of the corresponding - * function. - */ -/* Required for all the functions in this section */ -//#define MBEDTLS_ECP_INTERNAL_ALT -/* Turn off software fallback for curves not supported in hardware */ -//#define MBEDTLS_ECP_NO_FALLBACK -/* Support for Weierstrass curves with Jacobi representation */ -//#define MBEDTLS_ECP_RANDOMIZE_JAC_ALT -//#define MBEDTLS_ECP_ADD_MIXED_ALT -//#define MBEDTLS_ECP_DOUBLE_JAC_ALT -//#define MBEDTLS_ECP_NORMALIZE_JAC_MANY_ALT -//#define MBEDTLS_ECP_NORMALIZE_JAC_ALT -/* Support for curves with Montgomery arithmetic */ -//#define MBEDTLS_ECP_DOUBLE_ADD_MXZ_ALT -//#define MBEDTLS_ECP_RANDOMIZE_MXZ_ALT -//#define MBEDTLS_ECP_NORMALIZE_MXZ_ALT - -/** - * \def MBEDTLS_ENTROPY_HARDWARE_ALT - * - * Uncomment this macro to let mbed TLS use your own implementation of a - * hardware entropy collector. - * - * Your function must be called \c mbedtls_hardware_poll(), have the same - * prototype as declared in library/entropy_poll.h, and accept NULL as first - * argument. - * - * Uncomment to use your own hardware entropy collector. - */ -//#define MBEDTLS_ENTROPY_HARDWARE_ALT - -/** - * \def MBEDTLS_AES_ROM_TABLES - * - * Use precomputed AES tables stored in ROM. - * - * Uncomment this macro to use precomputed AES tables stored in ROM. - * Comment this macro to generate AES tables in RAM at runtime. - * - * Tradeoff: Using precomputed ROM tables reduces RAM usage by ~8kb - * (or ~2kb if \c MBEDTLS_AES_FEWER_TABLES is used) and reduces the - * initialization time before the first AES operation can be performed. - * It comes at the cost of additional ~8kb ROM use (resp. ~2kb if \c - * MBEDTLS_AES_FEWER_TABLES below is used), and potentially degraded - * performance if ROM access is slower than RAM access. - * - * This option is independent of \c MBEDTLS_AES_FEWER_TABLES. - * - */ -//#define MBEDTLS_AES_ROM_TABLES - -/** - * \def MBEDTLS_AES_FEWER_TABLES - * - * Use less ROM/RAM for AES tables. - * - * Uncommenting this macro omits 75% of the AES tables from - * ROM / RAM (depending on the value of \c MBEDTLS_AES_ROM_TABLES) - * by computing their values on the fly during operations - * (the tables are entry-wise rotations of one another). - * - * Tradeoff: Uncommenting this reduces the RAM / ROM footprint - * by ~6kb but at the cost of more arithmetic operations during - * runtime. Specifically, one has to compare 4 accesses within - * different tables to 4 accesses with additional arithmetic - * operations within the same table. The performance gain/loss - * depends on the system and memory details. - * - * This option is independent of \c MBEDTLS_AES_ROM_TABLES. - * - */ -//#define MBEDTLS_AES_FEWER_TABLES - -/** - * \def MBEDTLS_CAMELLIA_SMALL_MEMORY - * - * Use less ROM for the Camellia implementation (saves about 768 bytes). - * - * Uncomment this macro to use less memory for Camellia. - */ -//#define MBEDTLS_CAMELLIA_SMALL_MEMORY - -/** - * \def MBEDTLS_CIPHER_MODE_CBC - * - * Enable Cipher Block Chaining mode (CBC) for symmetric ciphers. - */ -#define MBEDTLS_CIPHER_MODE_CBC - -/** - * \def MBEDTLS_CIPHER_MODE_CFB - * - * Enable Cipher Feedback mode (CFB) for symmetric ciphers. - */ -#define MBEDTLS_CIPHER_MODE_CFB - -/** - * \def MBEDTLS_CIPHER_MODE_CTR - * - * Enable Counter Block Cipher mode (CTR) for symmetric ciphers. - */ -#define MBEDTLS_CIPHER_MODE_CTR - -/** - * \def MBEDTLS_CIPHER_MODE_OFB - * - * Enable Output Feedback mode (OFB) for symmetric ciphers. - */ -#define MBEDTLS_CIPHER_MODE_OFB - -/** - * \def MBEDTLS_CIPHER_MODE_XTS - * - * Enable Xor-encrypt-xor with ciphertext stealing mode (XTS) for AES. - */ -#define MBEDTLS_CIPHER_MODE_XTS - -/** - * \def MBEDTLS_CIPHER_NULL_CIPHER - * - * Enable NULL cipher. - * Warning: Only do so when you know what you are doing. This allows for - * encryption or channels without any security! - * - * To enable the following ciphersuites: - * MBEDTLS_TLS_ECDH_ECDSA_WITH_NULL_SHA - * MBEDTLS_TLS_ECDH_RSA_WITH_NULL_SHA - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_NULL_SHA - * MBEDTLS_TLS_ECDHE_RSA_WITH_NULL_SHA - * MBEDTLS_TLS_ECDHE_PSK_WITH_NULL_SHA384 - * MBEDTLS_TLS_ECDHE_PSK_WITH_NULL_SHA256 - * MBEDTLS_TLS_ECDHE_PSK_WITH_NULL_SHA - * MBEDTLS_TLS_DHE_PSK_WITH_NULL_SHA384 - * MBEDTLS_TLS_DHE_PSK_WITH_NULL_SHA256 - * MBEDTLS_TLS_DHE_PSK_WITH_NULL_SHA - * MBEDTLS_TLS_RSA_WITH_NULL_SHA256 - * MBEDTLS_TLS_RSA_WITH_NULL_SHA - * MBEDTLS_TLS_RSA_WITH_NULL_MD5 - * MBEDTLS_TLS_RSA_PSK_WITH_NULL_SHA384 - * MBEDTLS_TLS_RSA_PSK_WITH_NULL_SHA256 - * MBEDTLS_TLS_RSA_PSK_WITH_NULL_SHA - * MBEDTLS_TLS_PSK_WITH_NULL_SHA384 - * MBEDTLS_TLS_PSK_WITH_NULL_SHA256 - * MBEDTLS_TLS_PSK_WITH_NULL_SHA - * - * Uncomment this macro to enable the NULL cipher and ciphersuites - */ -//#define MBEDTLS_CIPHER_NULL_CIPHER - -/** - * \def MBEDTLS_CIPHER_PADDING_PKCS7 - * - * MBEDTLS_CIPHER_PADDING_XXX: Uncomment or comment macros to add support for - * specific padding modes in the cipher layer with cipher modes that support - * padding (e.g. CBC) - * - * If you disable all padding modes, only full blocks can be used with CBC. - * - * Enable padding modes in the cipher layer. - */ -#define MBEDTLS_CIPHER_PADDING_PKCS7 -#define MBEDTLS_CIPHER_PADDING_ONE_AND_ZEROS -#define MBEDTLS_CIPHER_PADDING_ZEROS_AND_LEN -#define MBEDTLS_CIPHER_PADDING_ZEROS - -/** \def MBEDTLS_CTR_DRBG_USE_128_BIT_KEY - * - * Uncomment this macro to use a 128-bit key in the CTR_DRBG module. - * By default, CTR_DRBG uses a 256-bit key. - */ -//#define MBEDTLS_CTR_DRBG_USE_128_BIT_KEY - -/** - * \def MBEDTLS_ECP_DP_SECP192R1_ENABLED - * - * MBEDTLS_ECP_XXXX_ENABLED: Enables specific curves within the Elliptic Curve - * module. By default all supported curves are enabled. - * - * Comment macros to disable the curve and functions for it - */ -/* Short Weierstrass curves (supporting ECP, ECDH, ECDSA) */ -#define MBEDTLS_ECP_DP_SECP192R1_ENABLED -#define MBEDTLS_ECP_DP_SECP224R1_ENABLED -#define MBEDTLS_ECP_DP_SECP256R1_ENABLED -#define MBEDTLS_ECP_DP_SECP384R1_ENABLED -#define MBEDTLS_ECP_DP_SECP521R1_ENABLED -#define MBEDTLS_ECP_DP_SECP192K1_ENABLED -#define MBEDTLS_ECP_DP_SECP224K1_ENABLED -#define MBEDTLS_ECP_DP_SECP256K1_ENABLED -#define MBEDTLS_ECP_DP_BP256R1_ENABLED -#define MBEDTLS_ECP_DP_BP384R1_ENABLED -#define MBEDTLS_ECP_DP_BP512R1_ENABLED -/* Montgomery curves (supporting ECP) */ -#define MBEDTLS_ECP_DP_CURVE25519_ENABLED -#define MBEDTLS_ECP_DP_CURVE448_ENABLED - -/** - * \def MBEDTLS_ECP_NIST_OPTIM - * - * Enable specific 'modulo p' routines for each NIST prime. - * Depending on the prime and architecture, makes operations 4 to 8 times - * faster on the corresponding curve. - * - * Comment this macro to disable NIST curves optimisation. - */ -#define MBEDTLS_ECP_NIST_OPTIM - -/** - * \def MBEDTLS_ECP_RESTARTABLE - * - * Enable "non-blocking" ECC operations that can return early and be resumed. - * - * This allows various functions to pause by returning - * #MBEDTLS_ERR_ECP_IN_PROGRESS (or, for functions in the SSL module, - * #MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS) and then be called later again in - * order to further progress and eventually complete their operation. This is - * controlled through mbedtls_ecp_set_max_ops() which limits the maximum - * number of ECC operations a function may perform before pausing; see - * mbedtls_ecp_set_max_ops() for more information. - * - * This is useful in non-threaded environments if you want to avoid blocking - * for too long on ECC (and, hence, X.509 or SSL/TLS) operations. - * - * Uncomment this macro to enable restartable ECC computations. - * - * \note This option only works with the default software implementation of - * elliptic curve functionality. It is incompatible with - * MBEDTLS_ECP_ALT, MBEDTLS_ECDH_XXX_ALT, MBEDTLS_ECDSA_XXX_ALT. - */ -//#define MBEDTLS_ECP_RESTARTABLE - -/** - * \def MBEDTLS_ECDSA_DETERMINISTIC - * - * Enable deterministic ECDSA (RFC 6979). - * Standard ECDSA is "fragile" in the sense that lack of entropy when signing - * may result in a compromise of the long-term signing key. This is avoided by - * the deterministic variant. - * - * Requires: MBEDTLS_HMAC_DRBG_C, MBEDTLS_ECDSA_C - * - * Comment this macro to disable deterministic ECDSA. - */ -#define MBEDTLS_ECDSA_DETERMINISTIC - -/** - * \def MBEDTLS_KEY_EXCHANGE_PSK_ENABLED - * - * Enable the PSK based ciphersuite modes in SSL / TLS. - * - * This enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_PSK_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_PSK_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_PSK_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_PSK_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_PSK_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_PSK_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_PSK_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_PSK_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_PSK_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_PSK_WITH_CAMELLIA_128_CBC_SHA256 - */ -#define MBEDTLS_KEY_EXCHANGE_PSK_ENABLED - -/** - * \def MBEDTLS_KEY_EXCHANGE_DHE_PSK_ENABLED - * - * Enable the DHE-PSK based ciphersuite modes in SSL / TLS. - * - * Requires: MBEDTLS_DHM_C - * - * This enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_128_CBC_SHA256 - * - * \warning Using DHE constitutes a security risk as it - * is not possible to validate custom DH parameters. - * If possible, it is recommended users should consider - * preferring other methods of key exchange. - * See dhm.h for more details. - * - */ -#define MBEDTLS_KEY_EXCHANGE_DHE_PSK_ENABLED - -/** - * \def MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED - * - * Enable the ECDHE-PSK based ciphersuite modes in SSL / TLS. - * - * Requires: MBEDTLS_ECDH_C - * - * This enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_ECDHE_PSK_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_ECDHE_PSK_WITH_CAMELLIA_128_CBC_SHA256 - */ -#define MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED - -/** - * \def MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED - * - * Enable the RSA-PSK based ciphersuite modes in SSL / TLS. - * - * Requires: MBEDTLS_RSA_C, MBEDTLS_PKCS1_V15, - * MBEDTLS_X509_CRT_PARSE_C - * - * This enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_128_CBC_SHA256 - */ -#define MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED - -/** - * \def MBEDTLS_KEY_EXCHANGE_RSA_ENABLED - * - * Enable the RSA-only based ciphersuite modes in SSL / TLS. - * - * Requires: MBEDTLS_RSA_C, MBEDTLS_PKCS1_V15, - * MBEDTLS_X509_CRT_PARSE_C - * - * This enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_RSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_RSA_WITH_AES_256_CBC_SHA256 - * MBEDTLS_TLS_RSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_CBC_SHA256 - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_CBC_SHA - * MBEDTLS_TLS_RSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_RSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_RSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_CBC_SHA - */ -#define MBEDTLS_KEY_EXCHANGE_RSA_ENABLED - -/** - * \def MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED - * - * Enable the DHE-RSA based ciphersuite modes in SSL / TLS. - * - * Requires: MBEDTLS_DHM_C, MBEDTLS_RSA_C, MBEDTLS_PKCS1_V15, - * MBEDTLS_X509_CRT_PARSE_C - * - * This enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CBC_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA - * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA - * - * \warning Using DHE constitutes a security risk as it - * is not possible to validate custom DH parameters. - * If possible, it is recommended users should consider - * preferring other methods of key exchange. - * See dhm.h for more details. - * - */ -#define MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED - -/** - * \def MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED - * - * Enable the ECDHE-RSA based ciphersuite modes in SSL / TLS. - * - * Requires: MBEDTLS_ECDH_C, MBEDTLS_RSA_C, MBEDTLS_PKCS1_V15, - * MBEDTLS_X509_CRT_PARSE_C - * - * This enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_128_CBC_SHA256 - */ -#define MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED - -/** - * \def MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED - * - * Enable the ECDHE-ECDSA based ciphersuite modes in SSL / TLS. - * - * Requires: MBEDTLS_ECDH_C, MBEDTLS_ECDSA_C, MBEDTLS_X509_CRT_PARSE_C, - * - * This enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_128_CBC_SHA256 - */ -#define MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED - -/** - * \def MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED - * - * Enable the ECDH-ECDSA based ciphersuite modes in SSL / TLS. - * - * Requires: MBEDTLS_ECDH_C, MBEDTLS_ECDSA_C, MBEDTLS_X509_CRT_PARSE_C - * - * This enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_256_GCM_SHA384 - */ -#define MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED - -/** - * \def MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED - * - * Enable the ECDH-RSA based ciphersuite modes in SSL / TLS. - * - * Requires: MBEDTLS_ECDH_C, MBEDTLS_RSA_C, MBEDTLS_X509_CRT_PARSE_C - * - * This enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_256_GCM_SHA384 - */ -#define MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED - -/** - * \def MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED - * - * Enable the ECJPAKE based ciphersuite modes in SSL / TLS. - * - * \warning This is currently experimental. EC J-PAKE support is based on the - * Thread v1.0.0 specification; incompatible changes to the specification - * might still happen. For this reason, this is disabled by default. - * - * Requires: MBEDTLS_ECJPAKE_C - * MBEDTLS_SHA256_C - * MBEDTLS_ECP_DP_SECP256R1_ENABLED - * - * This enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_ECJPAKE_WITH_AES_128_CCM_8 - */ -//#define MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED - -/** - * \def MBEDTLS_PK_PARSE_EC_EXTENDED - * - * Enhance support for reading EC keys using variants of SEC1 not allowed by - * RFC 5915 and RFC 5480. - * - * Currently this means parsing the SpecifiedECDomain choice of EC - * parameters (only known groups are supported, not arbitrary domains, to - * avoid validation issues). - * - * Disable if you only need to support RFC 5915 + 5480 key formats. - */ -#define MBEDTLS_PK_PARSE_EC_EXTENDED - -/** - * \def MBEDTLS_ERROR_STRERROR_DUMMY - * - * Enable a dummy error function to make use of mbedtls_strerror() in - * third party libraries easier when MBEDTLS_ERROR_C is disabled - * (no effect when MBEDTLS_ERROR_C is enabled). - * - * You can safely disable this if MBEDTLS_ERROR_C is enabled, or if you're - * not using mbedtls_strerror() or error_strerror() in your application. - * - * Disable if you run into name conflicts and want to really remove the - * mbedtls_strerror() - */ -#define MBEDTLS_ERROR_STRERROR_DUMMY - -/** - * \def MBEDTLS_GENPRIME - * - * Enable the prime-number generation code. - * - * Requires: MBEDTLS_BIGNUM_C - */ -#define MBEDTLS_GENPRIME - -/** - * \def MBEDTLS_FS_IO - * - * Enable functions that use the filesystem. - */ -#define MBEDTLS_FS_IO - -/** - * \def MBEDTLS_NO_DEFAULT_ENTROPY_SOURCES - * - * Do not add default entropy sources in mbedtls_entropy_init(). - * - * This is useful to have more control over the added entropy sources in an - * application. - * - * Uncomment this macro to prevent loading of default entropy functions. - */ -//#define MBEDTLS_NO_DEFAULT_ENTROPY_SOURCES - -/** - * \def MBEDTLS_NO_PLATFORM_ENTROPY - * - * Do not use built-in platform entropy functions. - * This is useful if your platform does not support - * standards like the /dev/urandom or Windows CryptoAPI. - * - * Uncomment this macro to disable the built-in platform entropy functions. - */ -//#define MBEDTLS_NO_PLATFORM_ENTROPY - -/** - * \def MBEDTLS_ENTROPY_FORCE_SHA256 - * - * Force the entropy accumulator to use a SHA-256 accumulator instead of the - * default SHA-512 based one (if both are available). - * - * Requires: MBEDTLS_SHA256_C - * - * On 32-bit systems SHA-256 can be much faster than SHA-512. Use this option - * if you have performance concerns. - * - * This option is only useful if both MBEDTLS_SHA256_C and - * MBEDTLS_SHA512_C are defined. Otherwise the available hash module is used. - */ -//#define MBEDTLS_ENTROPY_FORCE_SHA256 - -/** - * \def MBEDTLS_ENTROPY_NV_SEED - * - * Enable the non-volatile (NV) seed file-based entropy source. - * (Also enables the NV seed read/write functions in the platform layer) - * - * This is crucial (if not required) on systems that do not have a - * cryptographic entropy source (in hardware or kernel) available. - * - * Requires: MBEDTLS_ENTROPY_C, MBEDTLS_PLATFORM_C - * - * \note The read/write functions that are used by the entropy source are - * determined in the platform layer, and can be modified at runtime and/or - * compile-time depending on the flags (MBEDTLS_PLATFORM_NV_SEED_*) used. - * - * \note If you use the default implementation functions that read a seedfile - * with regular fopen(), please make sure you make a seedfile with the - * proper name (defined in MBEDTLS_PLATFORM_STD_NV_SEED_FILE) and at - * least MBEDTLS_ENTROPY_BLOCK_SIZE bytes in size that can be read from - * and written to or you will get an entropy source error! The default - * implementation will only use the first MBEDTLS_ENTROPY_BLOCK_SIZE - * bytes from the file. - * - * \note The entropy collector will write to the seed file before entropy is - * given to an external source, to update it. - */ -//#define MBEDTLS_ENTROPY_NV_SEED - -/* MBEDTLS_PSA_CRYPTO_KEY_ID_ENCODES_OWNER - * - * Enable key identifiers that encode a key owner identifier. - * - * The owner of a key is identified by a value of type ::mbedtls_key_owner_id_t - * which is currently hard-coded to be int32_t. - * - * Note that this option is meant for internal use only and may be removed - * without notice. It is incompatible with MBEDTLS_USE_PSA_CRYPTO. - */ -//#define MBEDTLS_PSA_CRYPTO_KEY_ID_ENCODES_OWNER - -/** - * \def MBEDTLS_MEMORY_DEBUG - * - * Enable debugging of buffer allocator memory issues. Automatically prints - * (to stderr) all (fatal) messages on memory allocation issues. Enables - * function for 'debug output' of allocated memory. - * - * Requires: MBEDTLS_MEMORY_BUFFER_ALLOC_C - * - * Uncomment this macro to let the buffer allocator print out error messages. - */ -//#define MBEDTLS_MEMORY_DEBUG - -/** - * \def MBEDTLS_MEMORY_BACKTRACE - * - * Include backtrace information with each allocated block. - * - * Requires: MBEDTLS_MEMORY_BUFFER_ALLOC_C - * GLIBC-compatible backtrace() an backtrace_symbols() support - * - * Uncomment this macro to include backtrace information - */ -//#define MBEDTLS_MEMORY_BACKTRACE - -/** - * \def MBEDTLS_PK_RSA_ALT_SUPPORT - * - * Support external private RSA keys (eg from a HSM) in the PK layer. - * - * Comment this macro to disable support for external private RSA keys. - */ -#define MBEDTLS_PK_RSA_ALT_SUPPORT - -/** - * \def MBEDTLS_PKCS1_V15 - * - * Enable support for PKCS#1 v1.5 encoding. - * - * Requires: MBEDTLS_RSA_C - * - * This enables support for PKCS#1 v1.5 operations. - */ -#define MBEDTLS_PKCS1_V15 - -/** - * \def MBEDTLS_PKCS1_V21 - * - * Enable support for PKCS#1 v2.1 encoding. - * - * Requires: MBEDTLS_MD_C, MBEDTLS_RSA_C - * - * This enables support for RSAES-OAEP and RSASSA-PSS operations. - */ -#define MBEDTLS_PKCS1_V21 - -/** \def MBEDTLS_PSA_CRYPTO_BUILTIN_KEYS - * - * Enable support for platform built-in keys. If you enable this feature, - * you must implement the function mbedtls_psa_platform_get_builtin_key(). - * See the documentation of that function for more information. - * - * Built-in keys are typically derived from a hardware unique key or - * stored in a secure element. - * - * Requires: MBEDTLS_PSA_CRYPTO_C. - * - * \warning This interface is experimental and may change or be removed - * without notice. - */ -//#define MBEDTLS_PSA_CRYPTO_BUILTIN_KEYS - -/** \def MBEDTLS_PSA_CRYPTO_CLIENT - * - * Enable support for PSA crypto client. - * - * \note This option allows to include the code necessary for a PSA - * crypto client when the PSA crypto implementation is not included in - * the library (MBEDTLS_PSA_CRYPTO_C disabled). The code included is the - * code to set and get PSA key attributes. - * The development of PSA drivers partially relying on the library to - * fulfill the hardware gaps is another possible usage of this option. - * - * \warning This interface is experimental and may change or be removed - * without notice. - */ -//#define MBEDTLS_PSA_CRYPTO_CLIENT - -/** \def MBEDTLS_PSA_CRYPTO_DRIVERS - * - * Enable support for the experimental PSA crypto driver interface. - * - * Requires: MBEDTLS_PSA_CRYPTO_C - * - * \warning This interface is experimental and may change or be removed - * without notice. - */ -//#define MBEDTLS_PSA_CRYPTO_DRIVERS - -/** \def MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG - * - * Make the PSA Crypto module use an external random generator provided - * by a driver, instead of Mbed TLS's entropy and DRBG modules. - * - * \note This random generator must deliver random numbers with cryptographic - * quality and high performance. It must supply unpredictable numbers - * with a uniform distribution. The implementation of this function - * is responsible for ensuring that the random generator is seeded - * with sufficient entropy. If you have a hardware TRNG which is slow - * or delivers non-uniform output, declare it as an entropy source - * with mbedtls_entropy_add_source() instead of enabling this option. - * - * If you enable this option, you must configure the type - * ::mbedtls_psa_external_random_context_t in psa/crypto_platform.h - * and define a function called mbedtls_psa_external_get_random() - * with the following prototype: - * ``` - * psa_status_t mbedtls_psa_external_get_random( - * mbedtls_psa_external_random_context_t *context, - * uint8_t *output, size_t output_size, size_t *output_length); - * ); - * ``` - * The \c context value is initialized to 0 before the first call. - * The function must fill the \c output buffer with \p output_size bytes - * of random data and set \c *output_length to \p output_size. - * - * Requires: MBEDTLS_PSA_CRYPTO_C - * - * \warning If you enable this option, code that uses the PSA cryptography - * interface will not use any of the entropy sources set up for - * the entropy module, nor the NV seed that MBEDTLS_ENTROPY_NV_SEED - * enables. - * - * \note This option is experimental and may be removed without notice. - */ -//#define MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG - -/** - * \def MBEDTLS_PSA_CRYPTO_SPM - * - * When MBEDTLS_PSA_CRYPTO_SPM is defined, the code is built for SPM (Secure - * Partition Manager) integration which separates the code into two parts: a - * NSPE (Non-Secure Process Environment) and an SPE (Secure Process - * Environment). - * - * Module: library/psa_crypto.c - * Requires: MBEDTLS_PSA_CRYPTO_C - * - */ -//#define MBEDTLS_PSA_CRYPTO_SPM - -/** - * \def MBEDTLS_PSA_INJECT_ENTROPY - * - * Enable support for entropy injection at first boot. This feature is - * required on systems that do not have a built-in entropy source (TRNG). - * This feature is currently not supported on systems that have a built-in - * entropy source. - * - * Requires: MBEDTLS_PSA_CRYPTO_STORAGE_C, MBEDTLS_ENTROPY_NV_SEED - * - */ -//#define MBEDTLS_PSA_INJECT_ENTROPY - -/** - * \def MBEDTLS_RSA_NO_CRT - * - * Do not use the Chinese Remainder Theorem - * for the RSA private operation. - * - * Uncomment this macro to disable the use of CRT in RSA. - * - */ -//#define MBEDTLS_RSA_NO_CRT - -/** - * \def MBEDTLS_SELF_TEST - * - * Enable the checkup functions (*_self_test). - */ -#define MBEDTLS_SELF_TEST - -/** - * \def MBEDTLS_SHA256_SMALLER - * - * Enable an implementation of SHA-256 that has lower ROM footprint but also - * lower performance. - * - * The default implementation is meant to be a reasonnable compromise between - * performance and size. This version optimizes more aggressively for size at - * the expense of performance. Eg on Cortex-M4 it reduces the size of - * mbedtls_sha256_process() from ~2KB to ~0.5KB for a performance hit of about - * 30%. - * - * Uncomment to enable the smaller implementation of SHA256. - */ -//#define MBEDTLS_SHA256_SMALLER - -/** - * \def MBEDTLS_SHA512_SMALLER - * - * Enable an implementation of SHA-512 that has lower ROM footprint but also - * lower performance. - * - * Uncomment to enable the smaller implementation of SHA512. - */ -//#define MBEDTLS_SHA512_SMALLER - -/** - * \def MBEDTLS_SSL_ALL_ALERT_MESSAGES - * - * Enable sending of alert messages in case of encountered errors as per RFC. - * If you choose not to send the alert messages, mbed TLS can still communicate - * with other servers, only debugging of failures is harder. - * - * The advantage of not sending alert messages, is that no information is given - * about reasons for failures thus preventing adversaries of gaining intel. - * - * Enable sending of all alert messages - */ -#define MBEDTLS_SSL_ALL_ALERT_MESSAGES - -/** - * \def MBEDTLS_SSL_DTLS_CONNECTION_ID - * - * Enable support for the DTLS Connection ID extension - * (version draft-ietf-tls-dtls-connection-id-05, - * https://tools.ietf.org/html/draft-ietf-tls-dtls-connection-id-05) - * which allows to identify DTLS connections across changes - * in the underlying transport. - * - * Setting this option enables the SSL APIs `mbedtls_ssl_set_cid()`, - * `mbedtls_ssl_get_peer_cid()` and `mbedtls_ssl_conf_cid()`. - * See the corresponding documentation for more information. - * - * \warning The Connection ID extension is still in draft state. - * We make no stability promises for the availability - * or the shape of the API controlled by this option. - * - * The maximum lengths of outgoing and incoming CIDs can be configured - * through the options - * - MBEDTLS_SSL_CID_OUT_LEN_MAX - * - MBEDTLS_SSL_CID_IN_LEN_MAX. - * - * Requires: MBEDTLS_SSL_PROTO_DTLS - * - * Uncomment to enable the Connection ID extension. - */ -//#define MBEDTLS_SSL_DTLS_CONNECTION_ID - -/** - * \def MBEDTLS_SSL_ASYNC_PRIVATE - * - * Enable asynchronous external private key operations in SSL. This allows - * you to configure an SSL connection to call an external cryptographic - * module to perform private key operations instead of performing the - * operation inside the library. - * - */ -//#define MBEDTLS_SSL_ASYNC_PRIVATE - -/** - * \def MBEDTLS_SSL_CONTEXT_SERIALIZATION - * - * Enable serialization of the TLS context structures, through use of the - * functions mbedtls_ssl_context_save() and mbedtls_ssl_context_load(). - * - * This pair of functions allows one side of a connection to serialize the - * context associated with the connection, then free or re-use that context - * while the serialized state is persisted elsewhere, and finally deserialize - * that state to a live context for resuming read/write operations on the - * connection. From a protocol perspective, the state of the connection is - * unaffected, in particular this is entirely transparent to the peer. - * - * Note: this is distinct from TLS session resumption, which is part of the - * protocol and fully visible by the peer. TLS session resumption enables - * establishing new connections associated to a saved session with shorter, - * lighter handshakes, while context serialization is a local optimization in - * handling a single, potentially long-lived connection. - * - * Enabling these APIs makes some SSL structures larger, as 64 extra bytes are - * saved after the handshake to allow for more efficient serialization, so if - * you don't need this feature you'll save RAM by disabling it. - * - * Comment to disable the context serialization APIs. - */ -#define MBEDTLS_SSL_CONTEXT_SERIALIZATION - -/** - * \def MBEDTLS_SSL_DEBUG_ALL - * - * Enable the debug messages in SSL module for all issues. - * Debug messages have been disabled in some places to prevent timing - * attacks due to (unbalanced) debugging function calls. - * - * If you need all error reporting you should enable this during debugging, - * but remove this for production servers that should log as well. - * - * Uncomment this macro to report all debug messages on errors introducing - * a timing side-channel. - * - */ -//#define MBEDTLS_SSL_DEBUG_ALL - -/** \def MBEDTLS_SSL_ENCRYPT_THEN_MAC - * - * Enable support for Encrypt-then-MAC, RFC 7366. - * - * This allows peers that both support it to use a more robust protection for - * ciphersuites using CBC, providing deep resistance against timing attacks - * on the padding or underlying cipher. - * - * This only affects CBC ciphersuites, and is useless if none is defined. - * - * Requires: MBEDTLS_SSL_PROTO_TLS1_2 - * - * Comment this macro to disable support for Encrypt-then-MAC - */ -#define MBEDTLS_SSL_ENCRYPT_THEN_MAC - -/** \def MBEDTLS_SSL_EXTENDED_MASTER_SECRET - * - * Enable support for RFC 7627: Session Hash and Extended Master Secret - * Extension. - * - * This was introduced as "the proper fix" to the Triple Handshake familiy of - * attacks, but it is recommended to always use it (even if you disable - * renegotiation), since it actually fixes a more fundamental issue in the - * original SSL/TLS design, and has implications beyond Triple Handshake. - * - * Requires: MBEDTLS_SSL_PROTO_TLS1_2 - * - * Comment this macro to disable support for Extended Master Secret. - */ -#define MBEDTLS_SSL_EXTENDED_MASTER_SECRET - -/** - * \def MBEDTLS_SSL_KEEP_PEER_CERTIFICATE - * - * This option controls the availability of the API mbedtls_ssl_get_peer_cert() - * giving access to the peer's certificate after completion of the handshake. - * - * Unless you need mbedtls_ssl_peer_cert() in your application, it is - * recommended to disable this option for reduced RAM usage. - * - * \note If this option is disabled, mbedtls_ssl_get_peer_cert() is still - * defined, but always returns \c NULL. - * - * \note This option has no influence on the protection against the - * triple handshake attack. Even if it is disabled, Mbed TLS will - * still ensure that certificates do not change during renegotiation, - * for exaple by keeping a hash of the peer's certificate. - * - * Comment this macro to disable storing the peer's certificate - * after the handshake. - */ -#define MBEDTLS_SSL_KEEP_PEER_CERTIFICATE - -/** - * \def MBEDTLS_SSL_RENEGOTIATION - * - * Enable support for TLS renegotiation. - * - * The two main uses of renegotiation are (1) refresh keys on long-lived - * connections and (2) client authentication after the initial handshake. - * If you don't need renegotiation, it's probably better to disable it, since - * it has been associated with security issues in the past and is easy to - * misuse/misunderstand. - * - * Comment this to disable support for renegotiation. - * - * \note Even if this option is disabled, both client and server are aware - * of the Renegotiation Indication Extension (RFC 5746) used to - * prevent the SSL renegotiation attack (see RFC 5746 Sect. 1). - * (See \c mbedtls_ssl_conf_legacy_renegotiation for the - * configuration of this extension). - * - */ -#define MBEDTLS_SSL_RENEGOTIATION - -/** - * \def MBEDTLS_SSL_MAX_FRAGMENT_LENGTH - * - * Enable support for RFC 6066 max_fragment_length extension in SSL. - * - * Comment this macro to disable support for the max_fragment_length extension - */ -#define MBEDTLS_SSL_MAX_FRAGMENT_LENGTH - -/** - * \def MBEDTLS_SSL_PROTO_TLS1_2 - * - * Enable support for TLS 1.2 (and DTLS 1.2 if DTLS is enabled). - * - * Requires: MBEDTLS_SHA1_C or MBEDTLS_SHA256_C or MBEDTLS_SHA512_C - * (Depends on ciphersuites) - * - * Comment this macro to disable support for TLS 1.2 / DTLS 1.2 - */ -#define MBEDTLS_SSL_PROTO_TLS1_2 - -/** - * \def MBEDTLS_SSL_PROTO_TLS1_3_EXPERIMENTAL - * - * This macro is used to selectively enable experimental parts - * of the code that contribute to the ongoing development of - * the prototype TLS 1.3 and DTLS 1.3 implementation, and provide - * no other purpose. - * - * \warning TLS 1.3 and DTLS 1.3 aren't yet supported in Mbed TLS, - * and no feature exposed through this macro is part of the - * public API. In particular, features under the control - * of this macro are experimental and don't come with any - * stability guarantees. - * - * Uncomment this macro to enable experimental and partial - * functionality specific to TLS 1.3. - */ -//#define MBEDTLS_SSL_PROTO_TLS1_3_EXPERIMENTAL - -/** - * \def MBEDTLS_SSL_PROTO_DTLS - * - * Enable support for DTLS (all available versions). - * - * Enable this and MBEDTLS_SSL_PROTO_TLS1_2 to enable DTLS 1.2. - * - * Requires: MBEDTLS_SSL_PROTO_TLS1_2 - * - * Comment this macro to disable support for DTLS - */ -#define MBEDTLS_SSL_PROTO_DTLS - -/** - * \def MBEDTLS_SSL_ALPN - * - * Enable support for RFC 7301 Application Layer Protocol Negotiation. - * - * Comment this macro to disable support for ALPN. - */ -#define MBEDTLS_SSL_ALPN - -/** - * \def MBEDTLS_SSL_DTLS_ANTI_REPLAY - * - * Enable support for the anti-replay mechanism in DTLS. - * - * Requires: MBEDTLS_SSL_TLS_C - * MBEDTLS_SSL_PROTO_DTLS - * - * \warning Disabling this is often a security risk! - * See mbedtls_ssl_conf_dtls_anti_replay() for details. - * - * Comment this to disable anti-replay in DTLS. - */ -#define MBEDTLS_SSL_DTLS_ANTI_REPLAY - -/** - * \def MBEDTLS_SSL_DTLS_HELLO_VERIFY - * - * Enable support for HelloVerifyRequest on DTLS servers. - * - * This feature is highly recommended to prevent DTLS servers being used as - * amplifiers in DoS attacks against other hosts. It should always be enabled - * unless you know for sure amplification cannot be a problem in the - * environment in which your server operates. - * - * \warning Disabling this can ba a security risk! (see above) - * - * Requires: MBEDTLS_SSL_PROTO_DTLS - * - * Comment this to disable support for HelloVerifyRequest. - */ -#define MBEDTLS_SSL_DTLS_HELLO_VERIFY - -/** - * \def MBEDTLS_SSL_DTLS_SRTP - * - * Enable support for negotiation of DTLS-SRTP (RFC 5764) - * through the use_srtp extension. - * - * \note This feature provides the minimum functionality required - * to negotiate the use of DTLS-SRTP and to allow the derivation of - * the associated SRTP packet protection key material. - * In particular, the SRTP packet protection itself, as well as the - * demultiplexing of RTP and DTLS packets at the datagram layer - * (see Section 5 of RFC 5764), are not handled by this feature. - * Instead, after successful completion of a handshake negotiating - * the use of DTLS-SRTP, the extended key exporter API - * mbedtls_ssl_conf_export_keys_cb() should be used to implement - * the key exporter described in Section 4.2 of RFC 5764 and RFC 5705 - * (this is implemented in the SSL example programs). - * The resulting key should then be passed to an SRTP stack. - * - * Setting this option enables the runtime API - * mbedtls_ssl_conf_dtls_srtp_protection_profiles() - * through which the supported DTLS-SRTP protection - * profiles can be configured. You must call this API at - * runtime if you wish to negotiate the use of DTLS-SRTP. - * - * Requires: MBEDTLS_SSL_PROTO_DTLS - * - * Uncomment this to enable support for use_srtp extension. - */ -//#define MBEDTLS_SSL_DTLS_SRTP - -/** - * \def MBEDTLS_SSL_DTLS_CLIENT_PORT_REUSE - * - * Enable server-side support for clients that reconnect from the same port. - * - * Some clients unexpectedly close the connection and try to reconnect using the - * same source port. This needs special support from the server to handle the - * new connection securely, as described in section 4.2.8 of RFC 6347. This - * flag enables that support. - * - * Requires: MBEDTLS_SSL_DTLS_HELLO_VERIFY - * - * Comment this to disable support for clients reusing the source port. - */ -#define MBEDTLS_SSL_DTLS_CLIENT_PORT_REUSE - -/** - * \def MBEDTLS_SSL_SESSION_TICKETS - * - * Enable support for RFC 5077 session tickets in SSL. - * Client-side, provides full support for session tickets (maintenance of a - * session store remains the responsibility of the application, though). - * Server-side, you also need to provide callbacks for writing and parsing - * tickets, including authenticated encryption and key management. Example - * callbacks are provided by MBEDTLS_SSL_TICKET_C. - * - * Comment this macro to disable support for SSL session tickets - */ -#define MBEDTLS_SSL_SESSION_TICKETS - -/** - * \def MBEDTLS_SSL_EXPORT_KEYS - * - * Enable support for exporting key block and master secret. - * This is required for certain users of TLS, e.g. EAP-TLS. - * - * Comment this macro to disable support for key export - */ -#define MBEDTLS_SSL_EXPORT_KEYS - -/** - * \def MBEDTLS_SSL_SERVER_NAME_INDICATION - * - * Enable support for RFC 6066 server name indication (SNI) in SSL. - * - * Requires: MBEDTLS_X509_CRT_PARSE_C - * - * Comment this macro to disable support for server name indication in SSL - */ -#define MBEDTLS_SSL_SERVER_NAME_INDICATION - -/** - * \def MBEDTLS_SSL_VARIABLE_BUFFER_LENGTH - * - * When this option is enabled, the SSL buffer will be resized automatically - * based on the negotiated maximum fragment length in each direction. - * - * Requires: MBEDTLS_SSL_MAX_FRAGMENT_LENGTH - */ -//#define MBEDTLS_SSL_VARIABLE_BUFFER_LENGTH - -/** - * \def MBEDTLS_TEST_CONSTANT_FLOW_MEMSAN - * - * Enable testing of the constant-flow nature of some sensitive functions with - * clang's MemorySanitizer. This causes some existing tests to also test - * this non-functional property of the code under test. - * - * This setting requires compiling with clang -fsanitize=memory. The test - * suites can then be run normally. - * - * \warning This macro is only used for extended testing; it is not considered - * part of the library's API, so it may change or disappear at any time. - * - * Uncomment to enable testing of the constant-flow nature of selected code. - */ -//#define MBEDTLS_TEST_CONSTANT_FLOW_MEMSAN - -/** - * \def MBEDTLS_TEST_CONSTANT_FLOW_VALGRIND - * - * Enable testing of the constant-flow nature of some sensitive functions with - * valgrind's memcheck tool. This causes some existing tests to also test - * this non-functional property of the code under test. - * - * This setting requires valgrind headers for building, and is only useful for - * testing if the tests suites are run with valgrind's memcheck. This can be - * done for an individual test suite with 'valgrind ./test_suite_xxx', or when - * using CMake, this can be done for all test suites with 'make memcheck'. - * - * \warning This macro is only used for extended testing; it is not considered - * part of the library's API, so it may change or disappear at any time. - * - * Uncomment to enable testing of the constant-flow nature of selected code. - */ -//#define MBEDTLS_TEST_CONSTANT_FLOW_VALGRIND - -/** - * \def MBEDTLS_TEST_HOOKS - * - * Enable features for invasive testing such as introspection functions and - * hooks for fault injection. This enables additional unit tests. - * - * Merely enabling this feature should not change the behavior of the product. - * It only adds new code, and new branching points where the default behavior - * is the same as when this feature is disabled. - * However, this feature increases the attack surface: there is an added - * risk of vulnerabilities, and more gadgets that can make exploits easier. - * Therefore this feature must never be enabled in production. - * - * See `docs/architecture/testing/mbed-crypto-invasive-testing.md` for more - * information. - * - * Uncomment to enable invasive tests. - */ -//#define MBEDTLS_TEST_HOOKS - -/** - * \def MBEDTLS_THREADING_ALT - * - * Provide your own alternate threading implementation. - * - * Requires: MBEDTLS_THREADING_C - * - * Uncomment this to allow your own alternate threading implementation. - */ -//#define MBEDTLS_THREADING_ALT - -/** - * \def MBEDTLS_THREADING_PTHREAD - * - * Enable the pthread wrapper layer for the threading layer. - * - * Requires: MBEDTLS_THREADING_C - * - * Uncomment this to enable pthread mutexes. - */ -//#define MBEDTLS_THREADING_PTHREAD - -/** - * \def MBEDTLS_USE_PSA_CRYPTO - * - * Make the X.509 and TLS library use PSA for cryptographic operations, and - * enable new APIs for using keys handled by PSA Crypto. - * - * \note Development of this option is currently in progress, and parts of Mbed - * TLS's X.509 and TLS modules are not ported to PSA yet. However, these parts - * will still continue to work as usual, so enabling this option should not - * break backwards compatibility. - * - * \warning The PSA Crypto API is in beta stage. While you're welcome to - * experiment using it, incompatible API changes are still possible, and some - * parts may not have reached the same quality as the rest of Mbed TLS yet. - * - * \warning This option enables new Mbed TLS APIs that are dependent on the - * PSA Crypto API, so can't come with the same stability guarantees as the - * rest of the Mbed TLS APIs. You're welcome to experiment with them, but for - * now, access to these APIs is opt-in (via enabling the present option), in - * order to clearly differentiate them from the stable Mbed TLS APIs. - * - * Requires: MBEDTLS_PSA_CRYPTO_C. - * - * Uncomment this to enable internal use of PSA Crypto and new associated APIs. - */ -//#define MBEDTLS_USE_PSA_CRYPTO - -/** - * \def MBEDTLS_PSA_CRYPTO_CONFIG - * - * This setting allows support for cryptographic mechanisms through the PSA - * API to be configured separately from support through the mbedtls API. - * - * Uncomment this to enable use of PSA Crypto configuration settings which - * can be found in include/psa/crypto_config.h. - * - * This feature is still experimental and is not ready for production since - * it is not completed. - */ -//#define MBEDTLS_PSA_CRYPTO_CONFIG - -/** - * \def MBEDTLS_VERSION_FEATURES - * - * Allow run-time checking of compile-time enabled features. Thus allowing users - * to check at run-time if the library is for instance compiled with threading - * support via mbedtls_version_check_feature(). - * - * Requires: MBEDTLS_VERSION_C - * - * Comment this to disable run-time checking and save ROM space - */ -#define MBEDTLS_VERSION_FEATURES - -/** - * \def MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK - * - * If set, this enables the X.509 API `mbedtls_x509_crt_verify_with_ca_cb()` - * and the SSL API `mbedtls_ssl_conf_ca_cb()` which allow users to configure - * the set of trusted certificates through a callback instead of a linked - * list. - * - * This is useful for example in environments where a large number of trusted - * certificates is present and storing them in a linked list isn't efficient - * enough, or when the set of trusted certificates changes frequently. - * - * See the documentation of `mbedtls_x509_crt_verify_with_ca_cb()` and - * `mbedtls_ssl_conf_ca_cb()` for more information. - * - * Uncomment to enable trusted certificate callbacks. - */ -//#define MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK - -/** - * \def MBEDTLS_X509_REMOVE_INFO - * - * Disable mbedtls_x509_*_info() and related APIs. - * - * Uncomment to omit mbedtls_x509_*_info(), as well as mbedtls_debug_print_crt() - * and other functions/constants only used by these functions, thus reducing - * the code footprint by several KB. - */ -//#define MBEDTLS_X509_REMOVE_INFO - -/** - * \def MBEDTLS_X509_RSASSA_PSS_SUPPORT - * - * Enable parsing and verification of X.509 certificates, CRLs and CSRS - * signed with RSASSA-PSS (aka PKCS#1 v2.1). - * - * Comment this macro to disallow using RSASSA-PSS in certificates. - */ -#define MBEDTLS_X509_RSASSA_PSS_SUPPORT -/* \} name SECTION: mbed TLS feature support */ - -/** - * \name SECTION: mbed TLS modules - * - * This section enables or disables entire modules in mbed TLS - * \{ - */ - -/** - * \def MBEDTLS_AESNI_C - * - * Enable AES-NI support on x86-64. - * - * Module: library/aesni.c - * Caller: library/aes.c - * - * Requires: MBEDTLS_HAVE_ASM - * - * This modules adds support for the AES-NI instructions on x86-64 - */ -#define MBEDTLS_AESNI_C - -/** - * \def MBEDTLS_AES_C - * - * Enable the AES block cipher. - * - * Module: library/aes.c - * Caller: library/cipher.c - * library/pem.c - * library/ctr_drbg.c - * - * This module enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_RSA_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_RSA_WITH_AES_256_CBC_SHA256 - * MBEDTLS_TLS_RSA_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_RSA_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_RSA_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_RSA_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_RSA_PSK_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_RSA_PSK_WITH_AES_128_CBC_SHA - * MBEDTLS_TLS_PSK_WITH_AES_256_GCM_SHA384 - * MBEDTLS_TLS_PSK_WITH_AES_256_CBC_SHA384 - * MBEDTLS_TLS_PSK_WITH_AES_256_CBC_SHA - * MBEDTLS_TLS_PSK_WITH_AES_128_GCM_SHA256 - * MBEDTLS_TLS_PSK_WITH_AES_128_CBC_SHA256 - * MBEDTLS_TLS_PSK_WITH_AES_128_CBC_SHA - * - * PEM_PARSE uses AES for decrypting encrypted keys. - */ -#define MBEDTLS_AES_C - -/** - * \def MBEDTLS_ASN1_PARSE_C - * - * Enable the generic ASN1 parser. - * - * Module: library/asn1.c - * Caller: library/x509.c - * library/dhm.c - * library/pkcs12.c - * library/pkcs5.c - * library/pkparse.c - */ -#define MBEDTLS_ASN1_PARSE_C - -/** - * \def MBEDTLS_ASN1_WRITE_C - * - * Enable the generic ASN1 writer. - * - * Module: library/asn1write.c - * Caller: library/ecdsa.c - * library/pkwrite.c - * library/x509_create.c - * library/x509write_crt.c - * library/x509write_csr.c - */ -#define MBEDTLS_ASN1_WRITE_C - -/** - * \def MBEDTLS_BASE64_C - * - * Enable the Base64 module. - * - * Module: library/base64.c - * Caller: library/pem.c - * - * This module is required for PEM support (required by X.509). - */ -#define MBEDTLS_BASE64_C - -/** - * \def MBEDTLS_BIGNUM_C - * - * Enable the multi-precision integer library. - * - * Module: library/bignum.c - * Caller: library/dhm.c - * library/ecp.c - * library/ecdsa.c - * library/rsa.c - * library/rsa_alt_helpers.c - * library/ssl_tls.c - * - * This module is required for RSA, DHM and ECC (ECDH, ECDSA) support. - */ -#define MBEDTLS_BIGNUM_C - -/** - * \def MBEDTLS_CAMELLIA_C - * - * Enable the Camellia block cipher. - * - * Module: library/camellia.c - * Caller: library/cipher.c - * - * This module enables the following ciphersuites (if other requisites are - * enabled as well): - * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA - * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_PSK_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_PSK_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_CBC_SHA256 - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_CBC_SHA - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_CBC_SHA - * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_128_CBC_SHA256 - * MBEDTLS_TLS_PSK_WITH_CAMELLIA_256_GCM_SHA384 - * MBEDTLS_TLS_PSK_WITH_CAMELLIA_256_CBC_SHA384 - * MBEDTLS_TLS_PSK_WITH_CAMELLIA_128_GCM_SHA256 - * MBEDTLS_TLS_PSK_WITH_CAMELLIA_128_CBC_SHA256 - */ -#define MBEDTLS_CAMELLIA_C - -/** - * \def MBEDTLS_ARIA_C - * - * Enable the ARIA block cipher. - * - * Module: library/aria.c - * Caller: library/cipher.c - * - * This module enables the following ciphersuites (if other requisites are - * enabled as well): - * - * MBEDTLS_TLS_RSA_WITH_ARIA_128_CBC_SHA256 - * MBEDTLS_TLS_RSA_WITH_ARIA_256_CBC_SHA384 - * MBEDTLS_TLS_DHE_RSA_WITH_ARIA_128_CBC_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_ARIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_128_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_128_CBC_SHA256 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_128_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_256_CBC_SHA384 - * MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_128_CBC_SHA256 - * MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_256_CBC_SHA384 - * MBEDTLS_TLS_RSA_WITH_ARIA_128_GCM_SHA256 - * MBEDTLS_TLS_RSA_WITH_ARIA_256_GCM_SHA384 - * MBEDTLS_TLS_DHE_RSA_WITH_ARIA_128_GCM_SHA256 - * MBEDTLS_TLS_DHE_RSA_WITH_ARIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_128_GCM_SHA256 - * MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_256_GCM_SHA384 - * MBEDTLS_TLS_PSK_WITH_ARIA_128_CBC_SHA256 - * MBEDTLS_TLS_PSK_WITH_ARIA_256_CBC_SHA384 - * MBEDTLS_TLS_DHE_PSK_WITH_ARIA_128_CBC_SHA256 - * MBEDTLS_TLS_DHE_PSK_WITH_ARIA_256_CBC_SHA384 - * MBEDTLS_TLS_RSA_PSK_WITH_ARIA_128_CBC_SHA256 - * MBEDTLS_TLS_RSA_PSK_WITH_ARIA_256_CBC_SHA384 - * MBEDTLS_TLS_PSK_WITH_ARIA_128_GCM_SHA256 - * MBEDTLS_TLS_PSK_WITH_ARIA_256_GCM_SHA384 - * MBEDTLS_TLS_DHE_PSK_WITH_ARIA_128_GCM_SHA256 - * MBEDTLS_TLS_DHE_PSK_WITH_ARIA_256_GCM_SHA384 - * MBEDTLS_TLS_RSA_PSK_WITH_ARIA_128_GCM_SHA256 - * MBEDTLS_TLS_RSA_PSK_WITH_ARIA_256_GCM_SHA384 - * MBEDTLS_TLS_ECDHE_PSK_WITH_ARIA_128_CBC_SHA256 - * MBEDTLS_TLS_ECDHE_PSK_WITH_ARIA_256_CBC_SHA384 - */ -#define MBEDTLS_ARIA_C - -/** - * \def MBEDTLS_CCM_C - * - * Enable the Counter with CBC-MAC (CCM) mode for 128-bit block cipher. - * - * Module: library/ccm.c - * - * Requires: MBEDTLS_AES_C or MBEDTLS_CAMELLIA_C - * - * This module enables the AES-CCM ciphersuites, if other requisites are - * enabled as well. - */ -#define MBEDTLS_CCM_C - -/** - * \def MBEDTLS_CHACHA20_C - * - * Enable the ChaCha20 stream cipher. - * - * Module: library/chacha20.c - */ -#define MBEDTLS_CHACHA20_C - -/** - * \def MBEDTLS_CHACHAPOLY_C - * - * Enable the ChaCha20-Poly1305 AEAD algorithm. - * - * Module: library/chachapoly.c - * - * This module requires: MBEDTLS_CHACHA20_C, MBEDTLS_POLY1305_C - */ -#define MBEDTLS_CHACHAPOLY_C - -/** - * \def MBEDTLS_CIPHER_C - * - * Enable the generic cipher layer. - * - * Module: library/cipher.c - * Caller: library/ssl_tls.c - * - * Uncomment to enable generic cipher wrappers. - */ -#define MBEDTLS_CIPHER_C - -/** - * \def MBEDTLS_CMAC_C - * - * Enable the CMAC (Cipher-based Message Authentication Code) mode for block - * ciphers. - * - * \note When #MBEDTLS_CMAC_ALT is active, meaning that the underlying - * implementation of the CMAC algorithm is provided by an alternate - * implementation, that alternate implementation may opt to not support - * AES-192 or 3DES as underlying block ciphers for the CMAC operation. - * - * Module: library/cmac.c - * - * Requires: MBEDTLS_AES_C or MBEDTLS_DES_C - * - */ -#define MBEDTLS_CMAC_C - -/** - * \def MBEDTLS_CTR_DRBG_C - * - * Enable the CTR_DRBG AES-based random generator. - * The CTR_DRBG generator uses AES-256 by default. - * To use AES-128 instead, enable \c MBEDTLS_CTR_DRBG_USE_128_BIT_KEY above. - * - * \note To achieve a 256-bit security strength with CTR_DRBG, - * you must use AES-256 *and* use sufficient entropy. - * See ctr_drbg.h for more details. - * - * Module: library/ctr_drbg.c - * Caller: - * - * Requires: MBEDTLS_AES_C - * - * This module provides the CTR_DRBG AES random number generator. - */ -#define MBEDTLS_CTR_DRBG_C - -/** - * \def MBEDTLS_DEBUG_C - * - * Enable the debug functions. - * - * Module: library/debug.c - * Caller: library/ssl_cli.c - * library/ssl_srv.c - * library/ssl_tls.c - * - * This module provides debugging functions. - */ -#define MBEDTLS_DEBUG_C - -/** - * \def MBEDTLS_DES_C - * - * Enable the DES block cipher. - * - * Module: library/des.c - * Caller: library/pem.c - * library/cipher.c - * - * PEM_PARSE uses DES/3DES for decrypting encrypted keys. - * - * \warning DES is considered a weak cipher and its use constitutes a - * security risk. We recommend considering stronger ciphers instead. - */ -#define MBEDTLS_DES_C - -/** - * \def MBEDTLS_DHM_C - * - * Enable the Diffie-Hellman-Merkle module. - * - * Module: library/dhm.c - * Caller: library/ssl_cli.c - * library/ssl_srv.c - * - * This module is used by the following key exchanges: - * DHE-RSA, DHE-PSK - * - * \warning Using DHE constitutes a security risk as it - * is not possible to validate custom DH parameters. - * If possible, it is recommended users should consider - * preferring other methods of key exchange. - * See dhm.h for more details. - * - */ -#define MBEDTLS_DHM_C - -/** - * \def MBEDTLS_ECDH_C - * - * Enable the elliptic curve Diffie-Hellman library. - * - * Module: library/ecdh.c - * Caller: library/ssl_cli.c - * library/ssl_srv.c - * - * This module is used by the following key exchanges: - * ECDHE-ECDSA, ECDHE-RSA, DHE-PSK - * - * Requires: MBEDTLS_ECP_C - */ -#define MBEDTLS_ECDH_C - -/** - * \def MBEDTLS_ECDSA_C - * - * Enable the elliptic curve DSA library. - * - * Module: library/ecdsa.c - * Caller: - * - * This module is used by the following key exchanges: - * ECDHE-ECDSA - * - * Requires: MBEDTLS_ECP_C, MBEDTLS_ASN1_WRITE_C, MBEDTLS_ASN1_PARSE_C, - * and at least one MBEDTLS_ECP_DP_XXX_ENABLED for a - * short Weierstrass curve. - */ -#define MBEDTLS_ECDSA_C - -/** - * \def MBEDTLS_ECJPAKE_C - * - * Enable the elliptic curve J-PAKE library. - * - * \note EC J-PAKE support is based on the Thread v1.0.0 specification. - * It has not been reviewed for compliance with newer standards such as - * Thread v1.1 or RFC 8236. - * - * Module: library/ecjpake.c - * Caller: - * - * This module is used by the following key exchanges: - * ECJPAKE - * - * Requires: MBEDTLS_ECP_C, MBEDTLS_MD_C - */ -#define MBEDTLS_ECJPAKE_C - -/** - * \def MBEDTLS_ECP_C - * - * Enable the elliptic curve over GF(p) library. - * - * Module: library/ecp.c - * Caller: library/ecdh.c - * library/ecdsa.c - * library/ecjpake.c - * - * Requires: MBEDTLS_BIGNUM_C and at least one MBEDTLS_ECP_DP_XXX_ENABLED - */ -#define MBEDTLS_ECP_C - -/** - * \def MBEDTLS_ENTROPY_C - * - * Enable the platform-specific entropy code. - * - * Module: library/entropy.c - * Caller: - * - * Requires: MBEDTLS_SHA512_C or MBEDTLS_SHA256_C - * - * This module provides a generic entropy pool - */ -#define MBEDTLS_ENTROPY_C - -/** - * \def MBEDTLS_ERROR_C - * - * Enable error code to error string conversion. - * - * Module: library/error.c - * Caller: - * - * This module enables mbedtls_strerror(). - */ -#define MBEDTLS_ERROR_C - -/** - * \def MBEDTLS_GCM_C - * - * Enable the Galois/Counter Mode (GCM). - * - * Module: library/gcm.c - * - * Requires: MBEDTLS_AES_C or MBEDTLS_CAMELLIA_C or MBEDTLS_ARIA_C - * - * This module enables the AES-GCM and CAMELLIA-GCM ciphersuites, if other - * requisites are enabled as well. - */ -#define MBEDTLS_GCM_C - -/** - * \def MBEDTLS_HKDF_C - * - * Enable the HKDF algorithm (RFC 5869). - * - * Module: library/hkdf.c - * Caller: - * - * Requires: MBEDTLS_MD_C - * - * This module adds support for the Hashed Message Authentication Code - * (HMAC)-based key derivation function (HKDF). - */ -#define MBEDTLS_HKDF_C - -/** - * \def MBEDTLS_HMAC_DRBG_C - * - * Enable the HMAC_DRBG random generator. - * - * Module: library/hmac_drbg.c - * Caller: - * - * Requires: MBEDTLS_MD_C - * - * Uncomment to enable the HMAC_DRBG random number geerator. - */ -#define MBEDTLS_HMAC_DRBG_C - -/** - * \def MBEDTLS_NIST_KW_C - * - * Enable the Key Wrapping mode for 128-bit block ciphers, - * as defined in NIST SP 800-38F. Only KW and KWP modes - * are supported. At the moment, only AES is approved by NIST. - * - * Module: library/nist_kw.c - * - * Requires: MBEDTLS_AES_C and MBEDTLS_CIPHER_C - */ -#define MBEDTLS_NIST_KW_C - -/** - * \def MBEDTLS_MD_C - * - * Enable the generic message digest layer. - * - * Module: library/md.c - * Caller: - * - * Uncomment to enable generic message digest wrappers. - */ -#define MBEDTLS_MD_C - -/** - * \def MBEDTLS_MD5_C - * - * Enable the MD5 hash algorithm. - * - * Module: library/md5.c - * Caller: library/md.c - * library/pem.c - * library/ssl_tls.c - * - * This module is required for TLS 1.2 depending on the handshake parameters. - * Further, it is used for checking MD5-signed certificates, and for PBKDF1 - * when decrypting PEM-encoded encrypted keys. - * - * \warning MD5 is considered a weak message digest and its use constitutes a - * security risk. If possible, we recommend avoiding dependencies on - * it, and considering stronger message digests instead. - * - */ -#define MBEDTLS_MD5_C - -/** - * \def MBEDTLS_MEMORY_BUFFER_ALLOC_C - * - * Enable the buffer allocator implementation that makes use of a (stack) - * based buffer to 'allocate' dynamic memory. (replaces calloc() and free() - * calls) - * - * Module: library/memory_buffer_alloc.c - * - * Requires: MBEDTLS_PLATFORM_C - * MBEDTLS_PLATFORM_MEMORY (to use it within mbed TLS) - * - * Enable this module to enable the buffer memory allocator. - */ -//#define MBEDTLS_MEMORY_BUFFER_ALLOC_C - -/** - * \def MBEDTLS_NET_C - * - * Enable the TCP and UDP over IPv6/IPv4 networking routines. - * - * \note This module only works on POSIX/Unix (including Linux, BSD and OS X) - * and Windows. For other platforms, you'll want to disable it, and write your - * own networking callbacks to be passed to \c mbedtls_ssl_set_bio(). - * - * \note See also our Knowledge Base article about porting to a new - * environment: - * https://tls.mbed.org/kb/how-to/how-do-i-port-mbed-tls-to-a-new-environment-OS - * - * Module: library/net_sockets.c - * - * This module provides networking routines. - */ -#define MBEDTLS_NET_C - -/** - * \def MBEDTLS_OID_C - * - * Enable the OID database. - * - * Module: library/oid.c - * Caller: library/asn1write.c - * library/pkcs5.c - * library/pkparse.c - * library/pkwrite.c - * library/rsa.c - * library/x509.c - * library/x509_create.c - * library/x509_crl.c - * library/x509_crt.c - * library/x509_csr.c - * library/x509write_crt.c - * library/x509write_csr.c - * - * This modules translates between OIDs and internal values. - */ -#define MBEDTLS_OID_C - -/** - * \def MBEDTLS_PADLOCK_C - * - * Enable VIA Padlock support on x86. - * - * Module: library/padlock.c - * Caller: library/aes.c - * - * Requires: MBEDTLS_HAVE_ASM - * - * This modules adds support for the VIA PadLock on x86. - */ -#define MBEDTLS_PADLOCK_C - -/** - * \def MBEDTLS_PEM_PARSE_C - * - * Enable PEM decoding / parsing. - * - * Module: library/pem.c - * Caller: library/dhm.c - * library/pkparse.c - * library/x509_crl.c - * library/x509_crt.c - * library/x509_csr.c - * - * Requires: MBEDTLS_BASE64_C - * - * This modules adds support for decoding / parsing PEM files. - */ -#define MBEDTLS_PEM_PARSE_C - -/** - * \def MBEDTLS_PEM_WRITE_C - * - * Enable PEM encoding / writing. - * - * Module: library/pem.c - * Caller: library/pkwrite.c - * library/x509write_crt.c - * library/x509write_csr.c - * - * Requires: MBEDTLS_BASE64_C - * - * This modules adds support for encoding / writing PEM files. - */ -#define MBEDTLS_PEM_WRITE_C - -/** - * \def MBEDTLS_PK_C - * - * Enable the generic public (asymetric) key layer. - * - * Module: library/pk.c - * Caller: library/ssl_tls.c - * library/ssl_cli.c - * library/ssl_srv.c - * - * Requires: MBEDTLS_RSA_C or MBEDTLS_ECP_C - * - * Uncomment to enable generic public key wrappers. - */ -#define MBEDTLS_PK_C - -/** - * \def MBEDTLS_PK_PARSE_C - * - * Enable the generic public (asymetric) key parser. - * - * Module: library/pkparse.c - * Caller: library/x509_crt.c - * library/x509_csr.c - * - * Requires: MBEDTLS_PK_C - * - * Uncomment to enable generic public key parse functions. - */ -#define MBEDTLS_PK_PARSE_C - -/** - * \def MBEDTLS_PK_WRITE_C - * - * Enable the generic public (asymetric) key writer. - * - * Module: library/pkwrite.c - * Caller: library/x509write.c - * - * Requires: MBEDTLS_PK_C - * - * Uncomment to enable generic public key write functions. - */ -#define MBEDTLS_PK_WRITE_C - -/** - * \def MBEDTLS_PKCS5_C - * - * Enable PKCS#5 functions. - * - * Module: library/pkcs5.c - * - * Requires: MBEDTLS_MD_C - * - * This module adds support for the PKCS#5 functions. - */ -#define MBEDTLS_PKCS5_C - -/** - * \def MBEDTLS_PKCS12_C - * - * Enable PKCS#12 PBE functions. - * Adds algorithms for parsing PKCS#8 encrypted private keys - * - * Module: library/pkcs12.c - * Caller: library/pkparse.c - * - * Requires: MBEDTLS_ASN1_PARSE_C, MBEDTLS_CIPHER_C, MBEDTLS_MD_C - * - * This module enables PKCS#12 functions. - */ -#define MBEDTLS_PKCS12_C - -/** - * \def MBEDTLS_PLATFORM_C - * - * Enable the platform abstraction layer that allows you to re-assign - * functions like calloc(), free(), snprintf(), printf(), fprintf(), exit(). - * - * Enabling MBEDTLS_PLATFORM_C enables to use of MBEDTLS_PLATFORM_XXX_ALT - * or MBEDTLS_PLATFORM_XXX_MACRO directives, allowing the functions mentioned - * above to be specified at runtime or compile time respectively. - * - * \note This abstraction layer must be enabled on Windows (including MSYS2) - * as other module rely on it for a fixed snprintf implementation. - * - * Module: library/platform.c - * Caller: Most other .c files - * - * This module enables abstraction of common (libc) functions. - */ -#define MBEDTLS_PLATFORM_C - -/** - * \def MBEDTLS_POLY1305_C - * - * Enable the Poly1305 MAC algorithm. - * - * Module: library/poly1305.c - * Caller: library/chachapoly.c - */ -#define MBEDTLS_POLY1305_C - -/** - * \def MBEDTLS_PSA_CRYPTO_C - * - * Enable the Platform Security Architecture cryptography API. - * - * \warning The PSA Crypto API is still beta status. While you're welcome to - * experiment using it, incompatible API changes are still possible, and some - * parts may not have reached the same quality as the rest of Mbed TLS yet. - * - * Module: library/psa_crypto.c - * - * Requires: either MBEDTLS_CTR_DRBG_C and MBEDTLS_ENTROPY_C, - * or MBEDTLS_HMAC_DRBG_C and MBEDTLS_ENTROPY_C, - * or MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG. - * - */ -#define MBEDTLS_PSA_CRYPTO_C - -/** - * \def MBEDTLS_PSA_CRYPTO_SE_C - * - * Enable secure element support in the Platform Security Architecture - * cryptography API. - * - * \warning This feature is not yet suitable for production. It is provided - * for API evaluation and testing purposes only. - * - * Module: library/psa_crypto_se.c - * - * Requires: MBEDTLS_PSA_CRYPTO_C, MBEDTLS_PSA_CRYPTO_STORAGE_C - * - */ -//#define MBEDTLS_PSA_CRYPTO_SE_C - -/** - * \def MBEDTLS_PSA_CRYPTO_STORAGE_C - * - * Enable the Platform Security Architecture persistent key storage. - * - * Module: library/psa_crypto_storage.c - * - * Requires: MBEDTLS_PSA_CRYPTO_C, - * either MBEDTLS_PSA_ITS_FILE_C or a native implementation of - * the PSA ITS interface - */ -#define MBEDTLS_PSA_CRYPTO_STORAGE_C - -/** - * \def MBEDTLS_PSA_ITS_FILE_C - * - * Enable the emulation of the Platform Security Architecture - * Internal Trusted Storage (PSA ITS) over files. - * - * Module: library/psa_its_file.c - * - * Requires: MBEDTLS_FS_IO - */ -#define MBEDTLS_PSA_ITS_FILE_C - -/** - * \def MBEDTLS_RIPEMD160_C - * - * Enable the RIPEMD-160 hash algorithm. - * - * Module: library/ripemd160.c - * Caller: library/md.c - * - */ -#define MBEDTLS_RIPEMD160_C - -/** - * \def MBEDTLS_RSA_C - * - * Enable the RSA public-key cryptosystem. - * - * Module: library/rsa.c - * library/rsa_alt_helpers.c - * Caller: library/ssl_cli.c - * library/ssl_srv.c - * library/ssl_tls.c - * library/x509.c - * - * This module is used by the following key exchanges: - * RSA, DHE-RSA, ECDHE-RSA, RSA-PSK - * - * Requires: MBEDTLS_BIGNUM_C, MBEDTLS_OID_C - */ -#define MBEDTLS_RSA_C - -/** - * \def MBEDTLS_SHA1_C - * - * Enable the SHA1 cryptographic hash algorithm. - * - * Module: library/sha1.c - * Caller: library/md.c - * library/ssl_cli.c - * library/ssl_srv.c - * library/ssl_tls.c - * library/x509write_crt.c - * - * This module is required for TLS 1.2 depending on the handshake parameters, - * and for SHA1-signed certificates. - * - * \warning SHA-1 is considered a weak message digest and its use constitutes - * a security risk. If possible, we recommend avoiding dependencies - * on it, and considering stronger message digests instead. - * - */ -#define MBEDTLS_SHA1_C - -/** - * \def MBEDTLS_SHA224_C - * - * Enable the SHA-224 cryptographic hash algorithm. - * - * Requires: MBEDTLS_SHA256_C. The library does not currently support enabling - * SHA-224 without SHA-256. - * - * Module: library/sha256.c - * Caller: library/md.c - * library/ssl_cookie.c - * - * This module adds support for SHA-224. - */ -#define MBEDTLS_SHA224_C - -/** - * \def MBEDTLS_SHA256_C - * - * Enable the SHA-256 cryptographic hash algorithm. - * - * Requires: MBEDTLS_SHA224_C. The library does not currently support enabling - * SHA-256 without SHA-224. - * - * Module: library/sha256.c - * Caller: library/entropy.c - * library/md.c - * library/ssl_cli.c - * library/ssl_srv.c - * library/ssl_tls.c - * - * This module adds support for SHA-256. - * This module is required for the SSL/TLS 1.2 PRF function. - */ -#define MBEDTLS_SHA256_C - -/** - * \def MBEDTLS_SHA384_C - * - * Enable the SHA-384 cryptographic hash algorithm. - * - * Requires: MBEDTLS_SHA512_C - * - * Module: library/sha512.c - * Caller: library/md.c - * library/ssl_cli.c - * library/ssl_srv.c - * - * Comment to disable SHA-384 - */ -#define MBEDTLS_SHA384_C - -/** - * \def MBEDTLS_SHA512_C - * - * Enable SHA-512 cryptographic hash algorithms. - * - * Module: library/sha512.c - * Caller: library/entropy.c - * library/md.c - * library/ssl_tls.c - * library/ssl_cookie.c - * - * This module adds support for SHA-512. - */ -#define MBEDTLS_SHA512_C - -/** - * \def MBEDTLS_SSL_CACHE_C - * - * Enable simple SSL cache implementation. - * - * Module: library/ssl_cache.c - * Caller: - * - * Requires: MBEDTLS_SSL_CACHE_C - */ -#define MBEDTLS_SSL_CACHE_C - -/** - * \def MBEDTLS_SSL_COOKIE_C - * - * Enable basic implementation of DTLS cookies for hello verification. - * - * Module: library/ssl_cookie.c - * Caller: - */ -#define MBEDTLS_SSL_COOKIE_C - -/** - * \def MBEDTLS_SSL_TICKET_C - * - * Enable an implementation of TLS server-side callbacks for session tickets. - * - * Module: library/ssl_ticket.c - * Caller: - * - * Requires: MBEDTLS_CIPHER_C - */ -#define MBEDTLS_SSL_TICKET_C - -/** - * \def MBEDTLS_SSL_CLI_C - * - * Enable the SSL/TLS client code. - * - * Module: library/ssl_cli.c - * Caller: - * - * Requires: MBEDTLS_SSL_TLS_C - * - * This module is required for SSL/TLS client support. - */ -#define MBEDTLS_SSL_CLI_C - -/** - * \def MBEDTLS_SSL_SRV_C - * - * Enable the SSL/TLS server code. - * - * Module: library/ssl_srv.c - * Caller: - * - * Requires: MBEDTLS_SSL_TLS_C - * - * This module is required for SSL/TLS server support. - */ -#define MBEDTLS_SSL_SRV_C - -/** - * \def MBEDTLS_SSL_TLS_C - * - * Enable the generic SSL/TLS code. - * - * Module: library/ssl_tls.c - * Caller: library/ssl_cli.c - * library/ssl_srv.c - * - * Requires: MBEDTLS_CIPHER_C, MBEDTLS_MD_C - * and at least one of the MBEDTLS_SSL_PROTO_XXX defines - * - * This module is required for SSL/TLS. - */ -#define MBEDTLS_SSL_TLS_C - -/** - * \def MBEDTLS_THREADING_C - * - * Enable the threading abstraction layer. - * By default mbed TLS assumes it is used in a non-threaded environment or that - * contexts are not shared between threads. If you do intend to use contexts - * between threads, you will need to enable this layer to prevent race - * conditions. See also our Knowledge Base article about threading: - * https://tls.mbed.org/kb/development/thread-safety-and-multi-threading - * - * Module: library/threading.c - * - * This allows different threading implementations (self-implemented or - * provided). - * - * You will have to enable either MBEDTLS_THREADING_ALT or - * MBEDTLS_THREADING_PTHREAD. - * - * Enable this layer to allow use of mutexes within mbed TLS - */ -//#define MBEDTLS_THREADING_C - -/** - * \def MBEDTLS_TIMING_C - * - * Enable the semi-portable timing interface. - * - * \note The provided implementation only works on POSIX/Unix (including Linux, - * BSD and OS X) and Windows. On other platforms, you can either disable that - * module and provide your own implementations of the callbacks needed by - * \c mbedtls_ssl_set_timer_cb() for DTLS, or leave it enabled and provide - * your own implementation of the whole module by setting - * \c MBEDTLS_TIMING_ALT in the current file. - * - * \note See also our Knowledge Base article about porting to a new - * environment: - * https://tls.mbed.org/kb/how-to/how-do-i-port-mbed-tls-to-a-new-environment-OS - * - * Module: library/timing.c - */ -#define MBEDTLS_TIMING_C - -/** - * \def MBEDTLS_VERSION_C - * - * Enable run-time version information. - * - * Module: library/version.c - * - * This module provides run-time version information. - */ -#define MBEDTLS_VERSION_C - -/** - * \def MBEDTLS_X509_USE_C - * - * Enable X.509 core for using certificates. - * - * Module: library/x509.c - * Caller: library/x509_crl.c - * library/x509_crt.c - * library/x509_csr.c - * - * Requires: MBEDTLS_ASN1_PARSE_C, MBEDTLS_BIGNUM_C, MBEDTLS_OID_C, - * MBEDTLS_PK_PARSE_C - * - * This module is required for the X.509 parsing modules. - */ -#define MBEDTLS_X509_USE_C - -/** - * \def MBEDTLS_X509_CRT_PARSE_C - * - * Enable X.509 certificate parsing. - * - * Module: library/x509_crt.c - * Caller: library/ssl_cli.c - * library/ssl_srv.c - * library/ssl_tls.c - * - * Requires: MBEDTLS_X509_USE_C - * - * This module is required for X.509 certificate parsing. - */ -#define MBEDTLS_X509_CRT_PARSE_C - -/** - * \def MBEDTLS_X509_CRL_PARSE_C - * - * Enable X.509 CRL parsing. - * - * Module: library/x509_crl.c - * Caller: library/x509_crt.c - * - * Requires: MBEDTLS_X509_USE_C - * - * This module is required for X.509 CRL parsing. - */ -#define MBEDTLS_X509_CRL_PARSE_C - -/** - * \def MBEDTLS_X509_CSR_PARSE_C - * - * Enable X.509 Certificate Signing Request (CSR) parsing. - * - * Module: library/x509_csr.c - * Caller: library/x509_crt_write.c - * - * Requires: MBEDTLS_X509_USE_C - * - * This module is used for reading X.509 certificate request. - */ -#define MBEDTLS_X509_CSR_PARSE_C - -/** - * \def MBEDTLS_X509_CREATE_C - * - * Enable X.509 core for creating certificates. - * - * Module: library/x509_create.c - * - * Requires: MBEDTLS_BIGNUM_C, MBEDTLS_OID_C, MBEDTLS_PK_WRITE_C - * - * This module is the basis for creating X.509 certificates and CSRs. - */ -#define MBEDTLS_X509_CREATE_C - -/** - * \def MBEDTLS_X509_CRT_WRITE_C - * - * Enable creating X.509 certificates. - * - * Module: library/x509_crt_write.c - * - * Requires: MBEDTLS_X509_CREATE_C - * - * This module is required for X.509 certificate creation. - */ -#define MBEDTLS_X509_CRT_WRITE_C - -/** - * \def MBEDTLS_X509_CSR_WRITE_C - * - * Enable creating X.509 Certificate Signing Requests (CSR). - * - * Module: library/x509_csr_write.c - * - * Requires: MBEDTLS_X509_CREATE_C - * - * This module is required for X.509 certificate request writing. - */ -#define MBEDTLS_X509_CSR_WRITE_C - -/* \} name SECTION: mbed TLS modules */ - -/** - * \name SECTION: Module configuration options - * - * This section allows for the setting of module specific sizes and - * configuration options. The default values are already present in the - * relevant header files and should suffice for the regular use cases. - * - * Our advice is to enable options and change their values here - * only if you have a good reason and know the consequences. - * - * Please check the respective header file for documentation on these - * parameters (to prevent duplicate documentation). - * \{ - */ - -/* MPI / BIGNUM options */ -//#define MBEDTLS_MPI_WINDOW_SIZE 6 /**< Maximum window size used. */ -//#define MBEDTLS_MPI_MAX_SIZE 1024 /**< Maximum number of bytes for usable MPIs. */ - -/* CTR_DRBG options */ -//#define MBEDTLS_CTR_DRBG_ENTROPY_LEN 48 /**< Amount of entropy used per seed by default (48 with SHA-512, 32 with SHA-256) */ -//#define MBEDTLS_CTR_DRBG_RESEED_INTERVAL 10000 /**< Interval before reseed is performed by default */ -//#define MBEDTLS_CTR_DRBG_MAX_INPUT 256 /**< Maximum number of additional input bytes */ -//#define MBEDTLS_CTR_DRBG_MAX_REQUEST 1024 /**< Maximum number of requested bytes per call */ -//#define MBEDTLS_CTR_DRBG_MAX_SEED_INPUT 384 /**< Maximum size of (re)seed buffer */ - -/* HMAC_DRBG options */ -//#define MBEDTLS_HMAC_DRBG_RESEED_INTERVAL 10000 /**< Interval before reseed is performed by default */ -//#define MBEDTLS_HMAC_DRBG_MAX_INPUT 256 /**< Maximum number of additional input bytes */ -//#define MBEDTLS_HMAC_DRBG_MAX_REQUEST 1024 /**< Maximum number of requested bytes per call */ -//#define MBEDTLS_HMAC_DRBG_MAX_SEED_INPUT 384 /**< Maximum size of (re)seed buffer */ - -/* ECP options */ -//#define MBEDTLS_ECP_WINDOW_SIZE 4 /**< Maximum window size used */ -//#define MBEDTLS_ECP_FIXED_POINT_OPTIM 1 /**< Enable fixed-point speed-up */ - -/* Entropy options */ -//#define MBEDTLS_ENTROPY_MAX_SOURCES 20 /**< Maximum number of sources supported */ -//#define MBEDTLS_ENTROPY_MAX_GATHER 128 /**< Maximum amount requested from entropy sources */ -//#define MBEDTLS_ENTROPY_MIN_HARDWARE 32 /**< Default minimum number of bytes required for the hardware entropy source mbedtls_hardware_poll() before entropy is released */ - -/* Memory buffer allocator options */ -//#define MBEDTLS_MEMORY_ALIGN_MULTIPLE 4 /**< Align on multiples of this value */ - -/* Platform options */ -//#define MBEDTLS_PLATFORM_STD_MEM_HDR /**< Header to include if MBEDTLS_PLATFORM_NO_STD_FUNCTIONS is defined. Don't define if no header is needed. */ -//#define MBEDTLS_PLATFORM_STD_CALLOC calloc /**< Default allocator to use, can be undefined */ -//#define MBEDTLS_PLATFORM_STD_FREE free /**< Default free to use, can be undefined */ -//#define MBEDTLS_PLATFORM_STD_EXIT exit /**< Default exit to use, can be undefined */ -//#define MBEDTLS_PLATFORM_STD_TIME time /**< Default time to use, can be undefined. MBEDTLS_HAVE_TIME must be enabled */ -//#define MBEDTLS_PLATFORM_STD_FPRINTF fprintf /**< Default fprintf to use, can be undefined */ -//#define MBEDTLS_PLATFORM_STD_PRINTF printf /**< Default printf to use, can be undefined */ -/* Note: your snprintf must correctly zero-terminate the buffer! */ -//#define MBEDTLS_PLATFORM_STD_SNPRINTF snprintf /**< Default snprintf to use, can be undefined */ -//#define MBEDTLS_PLATFORM_STD_EXIT_SUCCESS 0 /**< Default exit value to use, can be undefined */ -//#define MBEDTLS_PLATFORM_STD_EXIT_FAILURE 1 /**< Default exit value to use, can be undefined */ -//#define MBEDTLS_PLATFORM_STD_NV_SEED_READ mbedtls_platform_std_nv_seed_read /**< Default nv_seed_read function to use, can be undefined */ -//#define MBEDTLS_PLATFORM_STD_NV_SEED_WRITE mbedtls_platform_std_nv_seed_write /**< Default nv_seed_write function to use, can be undefined */ -//#define MBEDTLS_PLATFORM_STD_NV_SEED_FILE "seedfile" /**< Seed file to read/write with default implementation */ - -/* To Use Function Macros MBEDTLS_PLATFORM_C must be enabled */ -/* MBEDTLS_PLATFORM_XXX_MACRO and MBEDTLS_PLATFORM_XXX_ALT cannot both be defined */ -//#define MBEDTLS_PLATFORM_CALLOC_MACRO calloc /**< Default allocator macro to use, can be undefined */ -//#define MBEDTLS_PLATFORM_FREE_MACRO free /**< Default free macro to use, can be undefined */ -//#define MBEDTLS_PLATFORM_EXIT_MACRO exit /**< Default exit macro to use, can be undefined */ -//#define MBEDTLS_PLATFORM_TIME_MACRO time /**< Default time macro to use, can be undefined. MBEDTLS_HAVE_TIME must be enabled */ -//#define MBEDTLS_PLATFORM_TIME_TYPE_MACRO time_t /**< Default time macro to use, can be undefined. MBEDTLS_HAVE_TIME must be enabled */ -//#define MBEDTLS_PLATFORM_FPRINTF_MACRO fprintf /**< Default fprintf macro to use, can be undefined */ -//#define MBEDTLS_PLATFORM_PRINTF_MACRO printf /**< Default printf macro to use, can be undefined */ -/* Note: your snprintf must correctly zero-terminate the buffer! */ -//#define MBEDTLS_PLATFORM_SNPRINTF_MACRO snprintf /**< Default snprintf macro to use, can be undefined */ -//#define MBEDTLS_PLATFORM_VSNPRINTF_MACRO vsnprintf /**< Default vsnprintf macro to use, can be undefined */ -//#define MBEDTLS_PLATFORM_NV_SEED_READ_MACRO mbedtls_platform_std_nv_seed_read /**< Default nv_seed_read function to use, can be undefined */ -//#define MBEDTLS_PLATFORM_NV_SEED_WRITE_MACRO mbedtls_platform_std_nv_seed_write /**< Default nv_seed_write function to use, can be undefined */ - -/* PSA options */ -/** - * Use HMAC_DRBG with the specified hash algorithm for HMAC_DRBG for the - * PSA crypto subsystem. - * - * If this option is unset: - * - If CTR_DRBG is available, the PSA subsystem uses it rather than HMAC_DRBG. - * - Otherwise, the PSA subsystem uses HMAC_DRBG with either - * #MBEDTLS_MD_SHA512 or #MBEDTLS_MD_SHA256 based on availability and - * on unspecified heuristics. - */ -//#define MBEDTLS_PSA_HMAC_DRBG_MD_TYPE MBEDTLS_MD_SHA256 - -/** \def MBEDTLS_PSA_KEY_SLOT_COUNT - * Restrict the PSA library to supporting a maximum amount of simultaneously - * loaded keys. A loaded key is a key stored by the PSA Crypto core as a - * volatile key, or a persistent key which is loaded temporarily by the - * library as part of a crypto operation in flight. - * - * If this option is unset, the library will fall back to a default value of - * 32 keys. - */ -//#define MBEDTLS_PSA_KEY_SLOT_COUNT 32 - -/* SSL Cache options */ -//#define MBEDTLS_SSL_CACHE_DEFAULT_TIMEOUT 86400 /**< 1 day */ -//#define MBEDTLS_SSL_CACHE_DEFAULT_MAX_ENTRIES 50 /**< Maximum entries in cache */ - -/* SSL options */ - -/** \def MBEDTLS_SSL_IN_CONTENT_LEN - * - * Maximum length (in bytes) of incoming plaintext fragments. - * - * This determines the size of the incoming TLS I/O buffer in such a way - * that it is capable of holding the specified amount of plaintext data, - * regardless of the protection mechanism used. - * - * \note When using a value less than the default of 16KB on the client, it is - * recommended to use the Maximum Fragment Length (MFL) extension to - * inform the server about this limitation. On the server, there - * is no supported, standardized way of informing the client about - * restriction on the maximum size of incoming messages, and unless - * the limitation has been communicated by other means, it is recommended - * to only change the outgoing buffer size #MBEDTLS_SSL_OUT_CONTENT_LEN - * while keeping the default value of 16KB for the incoming buffer. - * - * Uncomment to set the maximum plaintext size of the incoming I/O buffer. - */ -//#define MBEDTLS_SSL_IN_CONTENT_LEN 16384 - -/** \def MBEDTLS_SSL_CID_IN_LEN_MAX - * - * The maximum length of CIDs used for incoming DTLS messages. - * - */ -//#define MBEDTLS_SSL_CID_IN_LEN_MAX 32 - -/** \def MBEDTLS_SSL_CID_OUT_LEN_MAX - * - * The maximum length of CIDs used for outgoing DTLS messages. - * - */ -//#define MBEDTLS_SSL_CID_OUT_LEN_MAX 32 - -/** \def MBEDTLS_SSL_CID_TLS1_3_PADDING_GRANULARITY - * - * This option controls the use of record plaintext padding - * in TLS 1.3 and when using the Connection ID extension in DTLS 1.2. - * - * The padding will always be chosen so that the length of the - * padded plaintext is a multiple of the value of this option. - * - * Note: A value of \c 1 means that no padding will be used - * for outgoing records. - * - * Note: On systems lacking division instructions, - * a power of two should be preferred. - */ -//#define MBEDTLS_SSL_CID_TLS1_3_PADDING_GRANULARITY 16 - -/** \def MBEDTLS_SSL_OUT_CONTENT_LEN - * - * Maximum length (in bytes) of outgoing plaintext fragments. - * - * This determines the size of the outgoing TLS I/O buffer in such a way - * that it is capable of holding the specified amount of plaintext data, - * regardless of the protection mechanism used. - * - * It is possible to save RAM by setting a smaller outward buffer, while keeping - * the default inward 16384 byte buffer to conform to the TLS specification. - * - * The minimum required outward buffer size is determined by the handshake - * protocol's usage. Handshaking will fail if the outward buffer is too small. - * The specific size requirement depends on the configured ciphers and any - * certificate data which is sent during the handshake. - * - * Uncomment to set the maximum plaintext size of the outgoing I/O buffer. - */ -//#define MBEDTLS_SSL_OUT_CONTENT_LEN 16384 - -/** \def MBEDTLS_SSL_DTLS_MAX_BUFFERING - * - * Maximum number of heap-allocated bytes for the purpose of - * DTLS handshake message reassembly and future message buffering. - * - * This should be at least 9/8 * MBEDTLSSL_IN_CONTENT_LEN - * to account for a reassembled handshake message of maximum size, - * together with its reassembly bitmap. - * - * A value of 2 * MBEDTLS_SSL_IN_CONTENT_LEN (32768 by default) - * should be sufficient for all practical situations as it allows - * to reassembly a large handshake message (such as a certificate) - * while buffering multiple smaller handshake messages. - * - */ -//#define MBEDTLS_SSL_DTLS_MAX_BUFFERING 32768 - -//#define MBEDTLS_PSK_MAX_LEN 32 /**< Max size of TLS pre-shared keys, in bytes (default 256 bits) */ -//#define MBEDTLS_SSL_COOKIE_TIMEOUT 60 /**< Default expiration delay of DTLS cookies, in seconds if HAVE_TIME, or in number of cookies issued */ - -/** - * Complete list of ciphersuites to use, in order of preference. - * - * \warning No dependency checking is done on that field! This option can only - * be used to restrict the set of available ciphersuites. It is your - * responsibility to make sure the needed modules are active. - * - * Use this to save a few hundred bytes of ROM (default ordering of all - * available ciphersuites) and a few to a few hundred bytes of RAM. - * - * The value below is only an example, not the default. - */ -//#define MBEDTLS_SSL_CIPHERSUITES MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384,MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 - -/* X509 options */ -//#define MBEDTLS_X509_MAX_INTERMEDIATE_CA 8 /**< Maximum number of intermediate CAs in a verification chain. */ -//#define MBEDTLS_X509_MAX_FILE_PATH_LEN 512 /**< Maximum length of a path/filename string in bytes including the null terminator character ('\0'). */ - -/** - * Uncomment the macro to let mbed TLS use your alternate implementation of - * mbedtls_platform_zeroize(). This replaces the default implementation in - * platform_util.c. - * - * mbedtls_platform_zeroize() is a widely used function across the library to - * zero a block of memory. The implementation is expected to be secure in the - * sense that it has been written to prevent the compiler from removing calls - * to mbedtls_platform_zeroize() as part of redundant code elimination - * optimizations. However, it is difficult to guarantee that calls to - * mbedtls_platform_zeroize() will not be optimized by the compiler as older - * versions of the C language standards do not provide a secure implementation - * of memset(). Therefore, MBEDTLS_PLATFORM_ZEROIZE_ALT enables users to - * configure their own implementation of mbedtls_platform_zeroize(), for - * example by using directives specific to their compiler, features from newer - * C standards (e.g using memset_s() in C11) or calling a secure memset() from - * their system (e.g explicit_bzero() in BSD). - */ -//#define MBEDTLS_PLATFORM_ZEROIZE_ALT - -/** - * Uncomment the macro to let Mbed TLS use your alternate implementation of - * mbedtls_platform_gmtime_r(). This replaces the default implementation in - * platform_util.c. - * - * gmtime() is not a thread-safe function as defined in the C standard. The - * library will try to use safer implementations of this function, such as - * gmtime_r() when available. However, if Mbed TLS cannot identify the target - * system, the implementation of mbedtls_platform_gmtime_r() will default to - * using the standard gmtime(). In this case, calls from the library to - * gmtime() will be guarded by the global mutex mbedtls_threading_gmtime_mutex - * if MBEDTLS_THREADING_C is enabled. We recommend that calls from outside the - * library are also guarded with this mutex to avoid race conditions. However, - * if the macro MBEDTLS_PLATFORM_GMTIME_R_ALT is defined, Mbed TLS will - * unconditionally use the implementation for mbedtls_platform_gmtime_r() - * supplied at compile time. - */ -//#define MBEDTLS_PLATFORM_GMTIME_R_ALT - -/** - * Enable the verified implementations of ECDH primitives from Project Everest - * (currently only Curve25519). This feature changes the layout of ECDH - * contexts and therefore is a compatibility break for applications that access - * fields of a mbedtls_ecdh_context structure directly. See also - * MBEDTLS_ECDH_LEGACY_CONTEXT in include/mbedtls/ecdh.h. - */ -//#define MBEDTLS_ECDH_VARIANT_EVEREST_ENABLED - -/* \} name SECTION: Customisation configuration options */ diff --git a/windows/mbedtls/md.h b/windows/mbedtls/md.h deleted file mode 100644 index 1170bc1a..00000000 --- a/windows/mbedtls/md.h +++ /dev/null @@ -1,446 +0,0 @@ - /** - * \file md.h - * - * \brief This file contains the generic message-digest wrapper. - * - * \author Adriaan de Jong - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_MD_H -#define MBEDTLS_MD_H -#include "mbedtls/private_access.h" - -#include - -#include "mbedtls/build_info.h" - -#define MBEDTLS_ERR_MD_FEATURE_UNAVAILABLE -0x5080 /**< The selected feature is not available. */ -#define MBEDTLS_ERR_MD_BAD_INPUT_DATA -0x5100 /**< Bad input parameters to function. */ -#define MBEDTLS_ERR_MD_ALLOC_FAILED -0x5180 /**< Failed to allocate memory. */ -#define MBEDTLS_ERR_MD_FILE_IO_ERROR -0x5200 /**< Opening or reading of file failed. */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief Supported message digests. - * - * \warning MD5 and SHA-1 are considered weak message digests and - * their use constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -typedef enum { - MBEDTLS_MD_NONE=0, /**< None. */ - MBEDTLS_MD_MD5, /**< The MD5 message digest. */ - MBEDTLS_MD_SHA1, /**< The SHA-1 message digest. */ - MBEDTLS_MD_SHA224, /**< The SHA-224 message digest. */ - MBEDTLS_MD_SHA256, /**< The SHA-256 message digest. */ - MBEDTLS_MD_SHA384, /**< The SHA-384 message digest. */ - MBEDTLS_MD_SHA512, /**< The SHA-512 message digest. */ - MBEDTLS_MD_RIPEMD160, /**< The RIPEMD-160 message digest. */ -} mbedtls_md_type_t; - -#if defined(MBEDTLS_SHA512_C) -#define MBEDTLS_MD_MAX_SIZE 64 /* longest known is SHA512 */ -#else -#define MBEDTLS_MD_MAX_SIZE 32 /* longest known is SHA256 or less */ -#endif - -#if defined(MBEDTLS_SHA512_C) -#define MBEDTLS_MD_MAX_BLOCK_SIZE 128 -#else -#define MBEDTLS_MD_MAX_BLOCK_SIZE 64 -#endif - -/** - * Opaque struct. - * - * Constructed using either #mbedtls_md_info_from_string or - * #mbedtls_md_info_from_type. - * - * Fields can be accessed with #mbedtls_md_get_size, - * #mbedtls_md_get_type and #mbedtls_md_get_name. - */ -/* Defined internally in library/md_wrap.h. */ -typedef struct mbedtls_md_info_t mbedtls_md_info_t; - -/** - * The generic message-digest context. - */ -typedef struct mbedtls_md_context_t -{ - /** Information about the associated message digest. */ - const mbedtls_md_info_t *MBEDTLS_PRIVATE(md_info); - - /** The digest-specific context. */ - void *MBEDTLS_PRIVATE(md_ctx); - - /** The HMAC part of the context. */ - void *MBEDTLS_PRIVATE(hmac_ctx); -} mbedtls_md_context_t; - -/** - * \brief This function returns the list of digests supported by the - * generic digest module. - * - * \note The list starts with the strongest available hashes. - * - * \return A statically allocated array of digests. Each element - * in the returned list is an integer belonging to the - * message-digest enumeration #mbedtls_md_type_t. - * The last entry is 0. - */ -const int *mbedtls_md_list( void ); - -/** - * \brief This function returns the message-digest information - * associated with the given digest name. - * - * \param md_name The name of the digest to search for. - * - * \return The message-digest information associated with \p md_name. - * \return NULL if the associated message-digest information is not found. - */ -const mbedtls_md_info_t *mbedtls_md_info_from_string( const char *md_name ); - -/** - * \brief This function returns the message-digest information - * associated with the given digest type. - * - * \param md_type The type of digest to search for. - * - * \return The message-digest information associated with \p md_type. - * \return NULL if the associated message-digest information is not found. - */ -const mbedtls_md_info_t *mbedtls_md_info_from_type( mbedtls_md_type_t md_type ); - -/** - * \brief This function initializes a message-digest context without - * binding it to a particular message-digest algorithm. - * - * This function should always be called first. It prepares the - * context for mbedtls_md_setup() for binding it to a - * message-digest algorithm. - */ -void mbedtls_md_init( mbedtls_md_context_t *ctx ); - -/** - * \brief This function clears the internal structure of \p ctx and - * frees any embedded internal structure, but does not free - * \p ctx itself. - * - * If you have called mbedtls_md_setup() on \p ctx, you must - * call mbedtls_md_free() when you are no longer using the - * context. - * Calling this function if you have previously - * called mbedtls_md_init() and nothing else is optional. - * You must not call this function if you have not called - * mbedtls_md_init(). - */ -void mbedtls_md_free( mbedtls_md_context_t *ctx ); - - -/** - * \brief This function selects the message digest algorithm to use, - * and allocates internal structures. - * - * It should be called after mbedtls_md_init() or - * mbedtls_md_free(). Makes it necessary to call - * mbedtls_md_free() later. - * - * \param ctx The context to set up. - * \param md_info The information structure of the message-digest algorithm - * to use. - * \param hmac Defines if HMAC is used. 0: HMAC is not used (saves some memory), - * or non-zero: HMAC is used with this context. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA on parameter-verification - * failure. - * \return #MBEDTLS_ERR_MD_ALLOC_FAILED on memory-allocation failure. - */ -int mbedtls_md_setup( mbedtls_md_context_t *ctx, const mbedtls_md_info_t *md_info, int hmac ); - -/** - * \brief This function clones the state of an message-digest - * context. - * - * \note You must call mbedtls_md_setup() on \c dst before calling - * this function. - * - * \note The two contexts must have the same type, - * for example, both are SHA-256. - * - * \warning This function clones the message-digest state, not the - * HMAC state. - * - * \param dst The destination context. - * \param src The context to be cloned. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA on parameter-verification failure. - */ -int mbedtls_md_clone( mbedtls_md_context_t *dst, - const mbedtls_md_context_t *src ); - -/** - * \brief This function extracts the message-digest size from the - * message-digest information structure. - * - * \param md_info The information structure of the message-digest algorithm - * to use. - * - * \return The size of the message-digest output in Bytes. - */ -unsigned char mbedtls_md_get_size( const mbedtls_md_info_t *md_info ); - -/** - * \brief This function extracts the message-digest type from the - * message-digest information structure. - * - * \param md_info The information structure of the message-digest algorithm - * to use. - * - * \return The type of the message digest. - */ -mbedtls_md_type_t mbedtls_md_get_type( const mbedtls_md_info_t *md_info ); - -/** - * \brief This function extracts the message-digest name from the - * message-digest information structure. - * - * \param md_info The information structure of the message-digest algorithm - * to use. - * - * \return The name of the message digest. - */ -const char *mbedtls_md_get_name( const mbedtls_md_info_t *md_info ); - -/** - * \brief This function starts a message-digest computation. - * - * You must call this function after setting up the context - * with mbedtls_md_setup(), and before passing data with - * mbedtls_md_update(). - * - * \param ctx The generic message-digest context. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA on parameter-verification - * failure. - */ -int mbedtls_md_starts( mbedtls_md_context_t *ctx ); - -/** - * \brief This function feeds an input buffer into an ongoing - * message-digest computation. - * - * You must call mbedtls_md_starts() before calling this - * function. You may call this function multiple times. - * Afterwards, call mbedtls_md_finish(). - * - * \param ctx The generic message-digest context. - * \param input The buffer holding the input data. - * \param ilen The length of the input data. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA on parameter-verification - * failure. - */ -int mbedtls_md_update( mbedtls_md_context_t *ctx, const unsigned char *input, size_t ilen ); - -/** - * \brief This function finishes the digest operation, - * and writes the result to the output buffer. - * - * Call this function after a call to mbedtls_md_starts(), - * followed by any number of calls to mbedtls_md_update(). - * Afterwards, you may either clear the context with - * mbedtls_md_free(), or call mbedtls_md_starts() to reuse - * the context for another digest operation with the same - * algorithm. - * - * \param ctx The generic message-digest context. - * \param output The buffer for the generic message-digest checksum result. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA on parameter-verification - * failure. - */ -int mbedtls_md_finish( mbedtls_md_context_t *ctx, unsigned char *output ); - -/** - * \brief This function calculates the message-digest of a buffer, - * with respect to a configurable message-digest algorithm - * in a single call. - * - * The result is calculated as - * Output = message_digest(input buffer). - * - * \param md_info The information structure of the message-digest algorithm - * to use. - * \param input The buffer holding the data. - * \param ilen The length of the input data. - * \param output The generic message-digest checksum result. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA on parameter-verification - * failure. - */ -int mbedtls_md( const mbedtls_md_info_t *md_info, const unsigned char *input, size_t ilen, - unsigned char *output ); - -#if defined(MBEDTLS_FS_IO) -/** - * \brief This function calculates the message-digest checksum - * result of the contents of the provided file. - * - * The result is calculated as - * Output = message_digest(file contents). - * - * \param md_info The information structure of the message-digest algorithm - * to use. - * \param path The input file name. - * \param output The generic message-digest checksum result. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_FILE_IO_ERROR on an I/O error accessing - * the file pointed by \p path. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA if \p md_info was NULL. - */ -int mbedtls_md_file( const mbedtls_md_info_t *md_info, const char *path, - unsigned char *output ); -#endif /* MBEDTLS_FS_IO */ - -/** - * \brief This function sets the HMAC key and prepares to - * authenticate a new message. - * - * Call this function after mbedtls_md_setup(), to use - * the MD context for an HMAC calculation, then call - * mbedtls_md_hmac_update() to provide the input data, and - * mbedtls_md_hmac_finish() to get the HMAC value. - * - * \param ctx The message digest context containing an embedded HMAC - * context. - * \param key The HMAC secret key. - * \param keylen The length of the HMAC key in Bytes. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA on parameter-verification - * failure. - */ -int mbedtls_md_hmac_starts( mbedtls_md_context_t *ctx, const unsigned char *key, - size_t keylen ); - -/** - * \brief This function feeds an input buffer into an ongoing HMAC - * computation. - * - * Call mbedtls_md_hmac_starts() or mbedtls_md_hmac_reset() - * before calling this function. - * You may call this function multiple times to pass the - * input piecewise. - * Afterwards, call mbedtls_md_hmac_finish(). - * - * \param ctx The message digest context containing an embedded HMAC - * context. - * \param input The buffer holding the input data. - * \param ilen The length of the input data. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA on parameter-verification - * failure. - */ -int mbedtls_md_hmac_update( mbedtls_md_context_t *ctx, const unsigned char *input, - size_t ilen ); - -/** - * \brief This function finishes the HMAC operation, and writes - * the result to the output buffer. - * - * Call this function after mbedtls_md_hmac_starts() and - * mbedtls_md_hmac_update() to get the HMAC value. Afterwards - * you may either call mbedtls_md_free() to clear the context, - * or call mbedtls_md_hmac_reset() to reuse the context with - * the same HMAC key. - * - * \param ctx The message digest context containing an embedded HMAC - * context. - * \param output The generic HMAC checksum result. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA on parameter-verification - * failure. - */ -int mbedtls_md_hmac_finish( mbedtls_md_context_t *ctx, unsigned char *output); - -/** - * \brief This function prepares to authenticate a new message with - * the same key as the previous HMAC operation. - * - * You may call this function after mbedtls_md_hmac_finish(). - * Afterwards call mbedtls_md_hmac_update() to pass the new - * input. - * - * \param ctx The message digest context containing an embedded HMAC - * context. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA on parameter-verification - * failure. - */ -int mbedtls_md_hmac_reset( mbedtls_md_context_t *ctx ); - -/** - * \brief This function calculates the full generic HMAC - * on the input buffer with the provided key. - * - * The function allocates the context, performs the - * calculation, and frees the context. - * - * The HMAC result is calculated as - * output = generic HMAC(hmac key, input buffer). - * - * \param md_info The information structure of the message-digest algorithm - * to use. - * \param key The HMAC secret key. - * \param keylen The length of the HMAC secret key in Bytes. - * \param input The buffer holding the input data. - * \param ilen The length of the input data. - * \param output The generic HMAC result. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MD_BAD_INPUT_DATA on parameter-verification - * failure. - */ -int mbedtls_md_hmac( const mbedtls_md_info_t *md_info, const unsigned char *key, size_t keylen, - const unsigned char *input, size_t ilen, - unsigned char *output ); - -/* Internal use */ -int mbedtls_md_process( mbedtls_md_context_t *ctx, const unsigned char *data ); - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_MD_H */ diff --git a/windows/mbedtls/md5.h b/windows/mbedtls/md5.h deleted file mode 100644 index e7befc34..00000000 --- a/windows/mbedtls/md5.h +++ /dev/null @@ -1,203 +0,0 @@ -/** - * \file md5.h - * - * \brief MD5 message digest algorithm (hash function) - * - * \warning MD5 is considered a weak message digest and its use constitutes a - * security risk. We recommend considering stronger message - * digests instead. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_MD5_H -#define MBEDTLS_MD5_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_MD5_ALT) -// Regular implementation -// - -/** - * \brief MD5 context structure - * - * \warning MD5 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -typedef struct mbedtls_md5_context -{ - uint32_t MBEDTLS_PRIVATE(total)[2]; /*!< number of bytes processed */ - uint32_t MBEDTLS_PRIVATE(state)[4]; /*!< intermediate digest state */ - unsigned char MBEDTLS_PRIVATE(buffer)[64]; /*!< data block being processed */ -} -mbedtls_md5_context; - -#else /* MBEDTLS_MD5_ALT */ -#include "md5_alt.h" -#endif /* MBEDTLS_MD5_ALT */ - -/** - * \brief Initialize MD5 context - * - * \param ctx MD5 context to be initialized - * - * \warning MD5 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -void mbedtls_md5_init( mbedtls_md5_context *ctx ); - -/** - * \brief Clear MD5 context - * - * \param ctx MD5 context to be cleared - * - * \warning MD5 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -void mbedtls_md5_free( mbedtls_md5_context *ctx ); - -/** - * \brief Clone (the state of) an MD5 context - * - * \param dst The destination context - * \param src The context to be cloned - * - * \warning MD5 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -void mbedtls_md5_clone( mbedtls_md5_context *dst, - const mbedtls_md5_context *src ); - -/** - * \brief MD5 context setup - * - * \param ctx context to be initialized - * - * \return 0 if successful - * - * \warning MD5 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -int mbedtls_md5_starts( mbedtls_md5_context *ctx ); - -/** - * \brief MD5 process buffer - * - * \param ctx MD5 context - * \param input buffer holding the data - * \param ilen length of the input data - * - * \return 0 if successful - * - * \warning MD5 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -int mbedtls_md5_update( mbedtls_md5_context *ctx, - const unsigned char *input, - size_t ilen ); - -/** - * \brief MD5 final digest - * - * \param ctx MD5 context - * \param output MD5 checksum result - * - * \return 0 if successful - * - * \warning MD5 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -int mbedtls_md5_finish( mbedtls_md5_context *ctx, - unsigned char output[16] ); - -/** - * \brief MD5 process data block (internal use only) - * - * \param ctx MD5 context - * \param data buffer holding one block of data - * - * \return 0 if successful - * - * \warning MD5 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -int mbedtls_internal_md5_process( mbedtls_md5_context *ctx, - const unsigned char data[64] ); - -/** - * \brief Output = MD5( input buffer ) - * - * \param input buffer holding the data - * \param ilen length of the input data - * \param output MD5 checksum result - * - * \return 0 if successful - * - * \warning MD5 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -int mbedtls_md5( const unsigned char *input, - size_t ilen, - unsigned char output[16] ); - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief Checkup routine - * - * \return 0 if successful, or 1 if the test failed - * - * \warning MD5 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -int mbedtls_md5_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* mbedtls_md5.h */ diff --git a/windows/mbedtls/memory_buffer_alloc.h b/windows/mbedtls/memory_buffer_alloc.h deleted file mode 100644 index d4737f5c..00000000 --- a/windows/mbedtls/memory_buffer_alloc.h +++ /dev/null @@ -1,145 +0,0 @@ -/** - * \file memory_buffer_alloc.h - * - * \brief Buffer-based memory allocator - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_MEMORY_BUFFER_ALLOC_H -#define MBEDTLS_MEMORY_BUFFER_ALLOC_H - -#include "mbedtls/build_info.h" - -#include - -/** - * \name SECTION: Module settings - * - * The configuration options you can set for this module are in this section. - * Either change them in mbedtls_config.h or define them on the compiler command line. - * \{ - */ - -#if !defined(MBEDTLS_MEMORY_ALIGN_MULTIPLE) -#define MBEDTLS_MEMORY_ALIGN_MULTIPLE 4 /**< Align on multiples of this value */ -#endif - -/* \} name SECTION: Module settings */ - -#define MBEDTLS_MEMORY_VERIFY_NONE 0 -#define MBEDTLS_MEMORY_VERIFY_ALLOC (1 << 0) -#define MBEDTLS_MEMORY_VERIFY_FREE (1 << 1) -#define MBEDTLS_MEMORY_VERIFY_ALWAYS (MBEDTLS_MEMORY_VERIFY_ALLOC | MBEDTLS_MEMORY_VERIFY_FREE) - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief Initialize use of stack-based memory allocator. - * The stack-based allocator does memory management inside the - * presented buffer and does not call calloc() and free(). - * It sets the global mbedtls_calloc() and mbedtls_free() pointers - * to its own functions. - * (Provided mbedtls_calloc() and mbedtls_free() are thread-safe if - * MBEDTLS_THREADING_C is defined) - * - * \note This code is not optimized and provides a straight-forward - * implementation of a stack-based memory allocator. - * - * \param buf buffer to use as heap - * \param len size of the buffer - */ -void mbedtls_memory_buffer_alloc_init( unsigned char *buf, size_t len ); - -/** - * \brief Free the mutex for thread-safety and clear remaining memory - */ -void mbedtls_memory_buffer_alloc_free( void ); - -/** - * \brief Determine when the allocator should automatically verify the state - * of the entire chain of headers / meta-data. - * (Default: MBEDTLS_MEMORY_VERIFY_NONE) - * - * \param verify One of MBEDTLS_MEMORY_VERIFY_NONE, MBEDTLS_MEMORY_VERIFY_ALLOC, - * MBEDTLS_MEMORY_VERIFY_FREE or MBEDTLS_MEMORY_VERIFY_ALWAYS - */ -void mbedtls_memory_buffer_set_verify( int verify ); - -#if defined(MBEDTLS_MEMORY_DEBUG) -/** - * \brief Print out the status of the allocated memory (primarily for use - * after a program should have de-allocated all memory) - * Prints out a list of 'still allocated' blocks and their stack - * trace if MBEDTLS_MEMORY_BACKTRACE is defined. - */ -void mbedtls_memory_buffer_alloc_status( void ); - -/** - * \brief Get the peak heap usage so far - * - * \param max_used Peak number of bytes in use or committed. This - * includes bytes in allocated blocks too small to split - * into smaller blocks but larger than the requested size. - * \param max_blocks Peak number of blocks in use, including free and used - */ -void mbedtls_memory_buffer_alloc_max_get( size_t *max_used, size_t *max_blocks ); - -/** - * \brief Reset peak statistics - */ -void mbedtls_memory_buffer_alloc_max_reset( void ); - -/** - * \brief Get the current heap usage - * - * \param cur_used Current number of bytes in use or committed. This - * includes bytes in allocated blocks too small to split - * into smaller blocks but larger than the requested size. - * \param cur_blocks Current number of blocks in use, including free and used - */ -void mbedtls_memory_buffer_alloc_cur_get( size_t *cur_used, size_t *cur_blocks ); -#endif /* MBEDTLS_MEMORY_DEBUG */ - -/** - * \brief Verifies that all headers in the memory buffer are correct - * and contain sane values. Helps debug buffer-overflow errors. - * - * Prints out first failure if MBEDTLS_MEMORY_DEBUG is defined. - * Prints out full header information if MBEDTLS_MEMORY_DEBUG - * is defined. (Includes stack trace information for each block if - * MBEDTLS_MEMORY_BACKTRACE is defined as well). - * - * \return 0 if verified, 1 otherwise - */ -int mbedtls_memory_buffer_alloc_verify( void ); - -#if defined(MBEDTLS_SELF_TEST) -/** - * \brief Checkup routine - * - * \return 0 if successful, or 1 if a test failed - */ -int mbedtls_memory_buffer_alloc_self_test( int verbose ); -#endif - -#ifdef __cplusplus -} -#endif - -#endif /* memory_buffer_alloc.h */ diff --git a/windows/mbedtls/net_sockets.h b/windows/mbedtls/net_sockets.h deleted file mode 100644 index 7b77329a..00000000 --- a/windows/mbedtls/net_sockets.h +++ /dev/null @@ -1,285 +0,0 @@ -/** - * \file net_sockets.h - * - * \brief Network sockets abstraction layer to integrate Mbed TLS into a - * BSD-style sockets API. - * - * The network sockets module provides an example integration of the - * Mbed TLS library into a BSD sockets implementation. The module is - * intended to be an example of how Mbed TLS can be integrated into a - * networking stack, as well as to be Mbed TLS's network integration - * for its supported platforms. - * - * The module is intended only to be used with the Mbed TLS library and - * is not intended to be used by third party application software - * directly. - * - * The supported platforms are as follows: - * * Microsoft Windows and Windows CE - * * POSIX/Unix platforms including Linux, OS X - * - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_NET_SOCKETS_H -#define MBEDTLS_NET_SOCKETS_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/ssl.h" - -#include -#include - -#define MBEDTLS_ERR_NET_SOCKET_FAILED -0x0042 /**< Failed to open a socket. */ -#define MBEDTLS_ERR_NET_CONNECT_FAILED -0x0044 /**< The connection to the given server / port failed. */ -#define MBEDTLS_ERR_NET_BIND_FAILED -0x0046 /**< Binding of the socket failed. */ -#define MBEDTLS_ERR_NET_LISTEN_FAILED -0x0048 /**< Could not listen on the socket. */ -#define MBEDTLS_ERR_NET_ACCEPT_FAILED -0x004A /**< Could not accept the incoming connection. */ -#define MBEDTLS_ERR_NET_RECV_FAILED -0x004C /**< Reading information from the socket failed. */ -#define MBEDTLS_ERR_NET_SEND_FAILED -0x004E /**< Sending information through the socket failed. */ -#define MBEDTLS_ERR_NET_CONN_RESET -0x0050 /**< Connection was reset by peer. */ -#define MBEDTLS_ERR_NET_UNKNOWN_HOST -0x0052 /**< Failed to get an IP address for the given hostname. */ -#define MBEDTLS_ERR_NET_BUFFER_TOO_SMALL -0x0043 /**< Buffer is too small to hold the data. */ -#define MBEDTLS_ERR_NET_INVALID_CONTEXT -0x0045 /**< The context is invalid, eg because it was free()ed. */ -#define MBEDTLS_ERR_NET_POLL_FAILED -0x0047 /**< Polling the net context failed. */ -#define MBEDTLS_ERR_NET_BAD_INPUT_DATA -0x0049 /**< Input invalid. */ - -#define MBEDTLS_NET_LISTEN_BACKLOG 10 /**< The backlog that listen() should use. */ - -#define MBEDTLS_NET_PROTO_TCP 0 /**< The TCP transport protocol */ -#define MBEDTLS_NET_PROTO_UDP 1 /**< The UDP transport protocol */ - -#define MBEDTLS_NET_POLL_READ 1 /**< Used in \c mbedtls_net_poll to check for pending data */ -#define MBEDTLS_NET_POLL_WRITE 2 /**< Used in \c mbedtls_net_poll to check if write possible */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * Wrapper type for sockets. - * - * Currently backed by just a file descriptor, but might be more in the future - * (eg two file descriptors for combined IPv4 + IPv6 support, or additional - * structures for hand-made UDP demultiplexing). - */ -typedef struct mbedtls_net_context -{ - int MBEDTLS_PRIVATE(fd); /**< The underlying file descriptor */ -} -mbedtls_net_context; - -/** - * \brief Initialize a context - * Just makes the context ready to be used or freed safely. - * - * \param ctx Context to initialize - */ -void mbedtls_net_init( mbedtls_net_context *ctx ); - -/** - * \brief Initiate a connection with host:port in the given protocol - * - * \param ctx Socket to use - * \param host Host to connect to - * \param port Port to connect to - * \param proto Protocol: MBEDTLS_NET_PROTO_TCP or MBEDTLS_NET_PROTO_UDP - * - * \return 0 if successful, or one of: - * MBEDTLS_ERR_NET_SOCKET_FAILED, - * MBEDTLS_ERR_NET_UNKNOWN_HOST, - * MBEDTLS_ERR_NET_CONNECT_FAILED - * - * \note Sets the socket in connected mode even with UDP. - */ -int mbedtls_net_connect( mbedtls_net_context *ctx, const char *host, const char *port, int proto ); - -/** - * \brief Create a receiving socket on bind_ip:port in the chosen - * protocol. If bind_ip == NULL, all interfaces are bound. - * - * \param ctx Socket to use - * \param bind_ip IP to bind to, can be NULL - * \param port Port number to use - * \param proto Protocol: MBEDTLS_NET_PROTO_TCP or MBEDTLS_NET_PROTO_UDP - * - * \return 0 if successful, or one of: - * MBEDTLS_ERR_NET_SOCKET_FAILED, - * MBEDTLS_ERR_NET_UNKNOWN_HOST, - * MBEDTLS_ERR_NET_BIND_FAILED, - * MBEDTLS_ERR_NET_LISTEN_FAILED - * - * \note Regardless of the protocol, opens the sockets and binds it. - * In addition, make the socket listening if protocol is TCP. - */ -int mbedtls_net_bind( mbedtls_net_context *ctx, const char *bind_ip, const char *port, int proto ); - -/** - * \brief Accept a connection from a remote client - * - * \param bind_ctx Relevant socket - * \param client_ctx Will contain the connected client socket - * \param client_ip Will contain the client IP address, can be NULL - * \param buf_size Size of the client_ip buffer - * \param ip_len Will receive the size of the client IP written, - * can be NULL if client_ip is null - * - * \return 0 if successful, or - * MBEDTLS_ERR_NET_SOCKET_FAILED, - * MBEDTLS_ERR_NET_BIND_FAILED, - * MBEDTLS_ERR_NET_ACCEPT_FAILED, or - * MBEDTLS_ERR_NET_BUFFER_TOO_SMALL if buf_size is too small, - * MBEDTLS_ERR_SSL_WANT_READ if bind_fd was set to - * non-blocking and accept() would block. - */ -int mbedtls_net_accept( mbedtls_net_context *bind_ctx, - mbedtls_net_context *client_ctx, - void *client_ip, size_t buf_size, size_t *ip_len ); - -/** - * \brief Check and wait for the context to be ready for read/write - * - * \note The current implementation of this function uses - * select() and returns an error if the file descriptor - * is \c FD_SETSIZE or greater. - * - * \param ctx Socket to check - * \param rw Bitflag composed of MBEDTLS_NET_POLL_READ and - * MBEDTLS_NET_POLL_WRITE specifying the events - * to wait for: - * - If MBEDTLS_NET_POLL_READ is set, the function - * will return as soon as the net context is available - * for reading. - * - If MBEDTLS_NET_POLL_WRITE is set, the function - * will return as soon as the net context is available - * for writing. - * \param timeout Maximal amount of time to wait before returning, - * in milliseconds. If \c timeout is zero, the - * function returns immediately. If \c timeout is - * -1u, the function blocks potentially indefinitely. - * - * \return Bitmask composed of MBEDTLS_NET_POLL_READ/WRITE - * on success or timeout, or a negative return code otherwise. - */ -int mbedtls_net_poll( mbedtls_net_context *ctx, uint32_t rw, uint32_t timeout ); - -/** - * \brief Set the socket blocking - * - * \param ctx Socket to set - * - * \return 0 if successful, or a non-zero error code - */ -int mbedtls_net_set_block( mbedtls_net_context *ctx ); - -/** - * \brief Set the socket non-blocking - * - * \param ctx Socket to set - * - * \return 0 if successful, or a non-zero error code - */ -int mbedtls_net_set_nonblock( mbedtls_net_context *ctx ); - -/** - * \brief Portable usleep helper - * - * \param usec Amount of microseconds to sleep - * - * \note Real amount of time slept will not be less than - * select()'s timeout granularity (typically, 10ms). - */ -void mbedtls_net_usleep( unsigned long usec ); - -/** - * \brief Read at most 'len' characters. If no error occurs, - * the actual amount read is returned. - * - * \param ctx Socket - * \param buf The buffer to write to - * \param len Maximum length of the buffer - * - * \return the number of bytes received, - * or a non-zero error code; with a non-blocking socket, - * MBEDTLS_ERR_SSL_WANT_READ indicates read() would block. - */ -int mbedtls_net_recv( void *ctx, unsigned char *buf, size_t len ); - -/** - * \brief Write at most 'len' characters. If no error occurs, - * the actual amount read is returned. - * - * \param ctx Socket - * \param buf The buffer to read from - * \param len The length of the buffer - * - * \return the number of bytes sent, - * or a non-zero error code; with a non-blocking socket, - * MBEDTLS_ERR_SSL_WANT_WRITE indicates write() would block. - */ -int mbedtls_net_send( void *ctx, const unsigned char *buf, size_t len ); - -/** - * \brief Read at most 'len' characters, blocking for at most - * 'timeout' seconds. If no error occurs, the actual amount - * read is returned. - * - * \note The current implementation of this function uses - * select() and returns an error if the file descriptor - * is \c FD_SETSIZE or greater. - * - * \param ctx Socket - * \param buf The buffer to write to - * \param len Maximum length of the buffer - * \param timeout Maximum number of milliseconds to wait for data - * 0 means no timeout (wait forever) - * - * \return The number of bytes received if successful. - * MBEDTLS_ERR_SSL_TIMEOUT if the operation timed out. - * MBEDTLS_ERR_SSL_WANT_READ if interrupted by a signal. - * Another negative error code (MBEDTLS_ERR_NET_xxx) - * for other failures. - * - * \note This function will block (until data becomes available or - * timeout is reached) even if the socket is set to - * non-blocking. Handling timeouts with non-blocking reads - * requires a different strategy. - */ -int mbedtls_net_recv_timeout( void *ctx, unsigned char *buf, size_t len, - uint32_t timeout ); - -/** - * \brief Closes down the connection and free associated data - * - * \param ctx The context to close - */ -void mbedtls_net_close( mbedtls_net_context *ctx ); - -/** - * \brief Gracefully shutdown the connection and free associated data - * - * \param ctx The context to free - */ -void mbedtls_net_free( mbedtls_net_context *ctx ); - -#ifdef __cplusplus -} -#endif - -#endif /* net_sockets.h */ diff --git a/windows/mbedtls/nist_kw.h b/windows/mbedtls/nist_kw.h deleted file mode 100644 index bd6c6584..00000000 --- a/windows/mbedtls/nist_kw.h +++ /dev/null @@ -1,179 +0,0 @@ -/** - * \file nist_kw.h - * - * \brief This file provides an API for key wrapping (KW) and key wrapping with - * padding (KWP) as defined in NIST SP 800-38F. - * https://nvlpubs.nist.gov/nistpubs/SpecialPublications/NIST.SP.800-38F.pdf - * - * Key wrapping specifies a deterministic authenticated-encryption mode - * of operation, according to NIST SP 800-38F: Recommendation for - * Block Cipher Modes of Operation: Methods for Key Wrapping. Its - * purpose is to protect cryptographic keys. - * - * Its equivalent is RFC 3394 for KW, and RFC 5649 for KWP. - * https://tools.ietf.org/html/rfc3394 - * https://tools.ietf.org/html/rfc5649 - * - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_NIST_KW_H -#define MBEDTLS_NIST_KW_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/cipher.h" - -#ifdef __cplusplus -extern "C" { -#endif - -typedef enum -{ - MBEDTLS_KW_MODE_KW = 0, - MBEDTLS_KW_MODE_KWP = 1 -} mbedtls_nist_kw_mode_t; - -#if !defined(MBEDTLS_NIST_KW_ALT) -// Regular implementation -// - -/** - * \brief The key wrapping context-type definition. The key wrapping context is passed - * to the APIs called. - * - * \note The definition of this type may change in future library versions. - * Don't make any assumptions on this context! - */ -typedef struct { - mbedtls_cipher_context_t MBEDTLS_PRIVATE(cipher_ctx); /*!< The cipher context used. */ -} mbedtls_nist_kw_context; - -#else /* MBEDTLS_NIST_key wrapping_ALT */ -#include "nist_kw_alt.h" -#endif /* MBEDTLS_NIST_KW_ALT */ - -/** - * \brief This function initializes the specified key wrapping context - * to make references valid and prepare the context - * for mbedtls_nist_kw_setkey() or mbedtls_nist_kw_free(). - * - * \param ctx The key wrapping context to initialize. - * - */ -void mbedtls_nist_kw_init( mbedtls_nist_kw_context *ctx ); - -/** - * \brief This function initializes the key wrapping context set in the - * \p ctx parameter and sets the encryption key. - * - * \param ctx The key wrapping context. - * \param cipher The 128-bit block cipher to use. Only AES is supported. - * \param key The Key Encryption Key (KEK). - * \param keybits The KEK size in bits. This must be acceptable by the cipher. - * \param is_wrap Specify whether the operation within the context is wrapping or unwrapping - * - * \return \c 0 on success. - * \return \c MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA for any invalid input. - * \return \c MBEDTLS_ERR_CIPHER_FEATURE_UNAVAILABLE for 128-bit block ciphers - * which are not supported. - * \return cipher-specific error code on failure of the underlying cipher. - */ -int mbedtls_nist_kw_setkey( mbedtls_nist_kw_context *ctx, - mbedtls_cipher_id_t cipher, - const unsigned char *key, - unsigned int keybits, - const int is_wrap ); - -/** - * \brief This function releases and clears the specified key wrapping context - * and underlying cipher sub-context. - * - * \param ctx The key wrapping context to clear. - */ -void mbedtls_nist_kw_free( mbedtls_nist_kw_context *ctx ); - -/** - * \brief This function encrypts a buffer using key wrapping. - * - * \param ctx The key wrapping context to use for encryption. - * \param mode The key wrapping mode to use (MBEDTLS_KW_MODE_KW or MBEDTLS_KW_MODE_KWP) - * \param input The buffer holding the input data. - * \param in_len The length of the input data in Bytes. - * The input uses units of 8 Bytes called semiblocks. - *
  • For KW mode: a multiple of 8 bytes between 16 and 2^57-8 inclusive.
  • - *
  • For KWP mode: any length between 1 and 2^32-1 inclusive.
- * \param[out] output The buffer holding the output data. - *
  • For KW mode: Must be at least 8 bytes larger than \p in_len.
  • - *
  • For KWP mode: Must be at least 8 bytes larger rounded up to a multiple of - * 8 bytes for KWP (15 bytes at most).
- * \param[out] out_len The number of bytes written to the output buffer. \c 0 on failure. - * \param[in] out_size The capacity of the output buffer. - * - * \return \c 0 on success. - * \return \c MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA for invalid input length. - * \return cipher-specific error code on failure of the underlying cipher. - */ -int mbedtls_nist_kw_wrap( mbedtls_nist_kw_context *ctx, mbedtls_nist_kw_mode_t mode, - const unsigned char *input, size_t in_len, - unsigned char *output, size_t* out_len, size_t out_size ); - -/** - * \brief This function decrypts a buffer using key wrapping. - * - * \param ctx The key wrapping context to use for decryption. - * \param mode The key wrapping mode to use (MBEDTLS_KW_MODE_KW or MBEDTLS_KW_MODE_KWP) - * \param input The buffer holding the input data. - * \param in_len The length of the input data in Bytes. - * The input uses units of 8 Bytes called semiblocks. - * The input must be a multiple of semiblocks. - *
  • For KW mode: a multiple of 8 bytes between 24 and 2^57 inclusive.
  • - *
  • For KWP mode: a multiple of 8 bytes between 16 and 2^32 inclusive.
- * \param[out] output The buffer holding the output data. - * The output buffer's minimal length is 8 bytes shorter than \p in_len. - * \param[out] out_len The number of bytes written to the output buffer. \c 0 on failure. - * For KWP mode, the length could be up to 15 bytes shorter than \p in_len, - * depending on how much padding was added to the data. - * \param[in] out_size The capacity of the output buffer. - * - * \return \c 0 on success. - * \return \c MBEDTLS_ERR_CIPHER_BAD_INPUT_DATA for invalid input length. - * \return \c MBEDTLS_ERR_CIPHER_AUTH_FAILED for verification failure of the ciphertext. - * \return cipher-specific error code on failure of the underlying cipher. - */ -int mbedtls_nist_kw_unwrap( mbedtls_nist_kw_context *ctx, mbedtls_nist_kw_mode_t mode, - const unsigned char *input, size_t in_len, - unsigned char *output, size_t* out_len, size_t out_size); - - -#if defined(MBEDTLS_SELF_TEST) && defined(MBEDTLS_AES_C) -/** - * \brief The key wrapping checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_nist_kw_self_test( int verbose ); -#endif /* MBEDTLS_SELF_TEST && MBEDTLS_AES_C */ - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_NIST_KW_H */ diff --git a/windows/mbedtls/oid.h b/windows/mbedtls/oid.h deleted file mode 100644 index 9e68e919..00000000 --- a/windows/mbedtls/oid.h +++ /dev/null @@ -1,641 +0,0 @@ -/** - * \file oid.h - * - * \brief Object Identifier (OID) database - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_OID_H -#define MBEDTLS_OID_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/asn1.h" -#include "mbedtls/pk.h" - -#include - -#if defined(MBEDTLS_CIPHER_C) -#include "mbedtls/cipher.h" -#endif - -#if defined(MBEDTLS_MD_C) -#include "mbedtls/md.h" -#endif - -#define MBEDTLS_ERR_OID_NOT_FOUND -0x002E /**< OID is not found. */ -#define MBEDTLS_ERR_OID_BUF_TOO_SMALL -0x000B /**< output buffer is too small */ - -/* This is for the benefit of X.509, but defined here in order to avoid - * having a "backwards" include of x.509.h here */ -/* - * X.509 extension types (internal, arbitrary values for bitsets) - */ -#define MBEDTLS_OID_X509_EXT_AUTHORITY_KEY_IDENTIFIER (1 << 0) -#define MBEDTLS_OID_X509_EXT_SUBJECT_KEY_IDENTIFIER (1 << 1) -#define MBEDTLS_OID_X509_EXT_KEY_USAGE (1 << 2) -#define MBEDTLS_OID_X509_EXT_CERTIFICATE_POLICIES (1 << 3) -#define MBEDTLS_OID_X509_EXT_POLICY_MAPPINGS (1 << 4) -#define MBEDTLS_OID_X509_EXT_SUBJECT_ALT_NAME (1 << 5) -#define MBEDTLS_OID_X509_EXT_ISSUER_ALT_NAME (1 << 6) -#define MBEDTLS_OID_X509_EXT_SUBJECT_DIRECTORY_ATTRS (1 << 7) -#define MBEDTLS_OID_X509_EXT_BASIC_CONSTRAINTS (1 << 8) -#define MBEDTLS_OID_X509_EXT_NAME_CONSTRAINTS (1 << 9) -#define MBEDTLS_OID_X509_EXT_POLICY_CONSTRAINTS (1 << 10) -#define MBEDTLS_OID_X509_EXT_EXTENDED_KEY_USAGE (1 << 11) -#define MBEDTLS_OID_X509_EXT_CRL_DISTRIBUTION_POINTS (1 << 12) -#define MBEDTLS_OID_X509_EXT_INIHIBIT_ANYPOLICY (1 << 13) -#define MBEDTLS_OID_X509_EXT_FRESHEST_CRL (1 << 14) -#define MBEDTLS_OID_X509_EXT_NS_CERT_TYPE (1 << 16) - -/* - * Top level OID tuples - */ -#define MBEDTLS_OID_ISO_MEMBER_BODIES "\x2a" /* {iso(1) member-body(2)} */ -#define MBEDTLS_OID_ISO_IDENTIFIED_ORG "\x2b" /* {iso(1) identified-organization(3)} */ -#define MBEDTLS_OID_ISO_CCITT_DS "\x55" /* {joint-iso-ccitt(2) ds(5)} */ -#define MBEDTLS_OID_ISO_ITU_COUNTRY "\x60" /* {joint-iso-itu-t(2) country(16)} */ - -/* - * ISO Member bodies OID parts - */ -#define MBEDTLS_OID_COUNTRY_US "\x86\x48" /* {us(840)} */ -#define MBEDTLS_OID_ORG_RSA_DATA_SECURITY "\x86\xf7\x0d" /* {rsadsi(113549)} */ -#define MBEDTLS_OID_RSA_COMPANY MBEDTLS_OID_ISO_MEMBER_BODIES MBEDTLS_OID_COUNTRY_US \ - MBEDTLS_OID_ORG_RSA_DATA_SECURITY /* {iso(1) member-body(2) us(840) rsadsi(113549)} */ -#define MBEDTLS_OID_ORG_ANSI_X9_62 "\xce\x3d" /* ansi-X9-62(10045) */ -#define MBEDTLS_OID_ANSI_X9_62 MBEDTLS_OID_ISO_MEMBER_BODIES MBEDTLS_OID_COUNTRY_US \ - MBEDTLS_OID_ORG_ANSI_X9_62 - -/* - * ISO Identified organization OID parts - */ -#define MBEDTLS_OID_ORG_DOD "\x06" /* {dod(6)} */ -#define MBEDTLS_OID_ORG_OIW "\x0e" -#define MBEDTLS_OID_OIW_SECSIG MBEDTLS_OID_ORG_OIW "\x03" -#define MBEDTLS_OID_OIW_SECSIG_ALG MBEDTLS_OID_OIW_SECSIG "\x02" -#define MBEDTLS_OID_OIW_SECSIG_SHA1 MBEDTLS_OID_OIW_SECSIG_ALG "\x1a" -#define MBEDTLS_OID_ORG_CERTICOM "\x81\x04" /* certicom(132) */ -#define MBEDTLS_OID_CERTICOM MBEDTLS_OID_ISO_IDENTIFIED_ORG MBEDTLS_OID_ORG_CERTICOM -#define MBEDTLS_OID_ORG_TELETRUST "\x24" /* teletrust(36) */ -#define MBEDTLS_OID_TELETRUST MBEDTLS_OID_ISO_IDENTIFIED_ORG MBEDTLS_OID_ORG_TELETRUST - -/* - * ISO ITU OID parts - */ -#define MBEDTLS_OID_ORGANIZATION "\x01" /* {organization(1)} */ -#define MBEDTLS_OID_ISO_ITU_US_ORG MBEDTLS_OID_ISO_ITU_COUNTRY MBEDTLS_OID_COUNTRY_US MBEDTLS_OID_ORGANIZATION /* {joint-iso-itu-t(2) country(16) us(840) organization(1)} */ - -#define MBEDTLS_OID_ORG_GOV "\x65" /* {gov(101)} */ -#define MBEDTLS_OID_GOV MBEDTLS_OID_ISO_ITU_US_ORG MBEDTLS_OID_ORG_GOV /* {joint-iso-itu-t(2) country(16) us(840) organization(1) gov(101)} */ - -#define MBEDTLS_OID_ORG_NETSCAPE "\x86\xF8\x42" /* {netscape(113730)} */ -#define MBEDTLS_OID_NETSCAPE MBEDTLS_OID_ISO_ITU_US_ORG MBEDTLS_OID_ORG_NETSCAPE /* Netscape OID {joint-iso-itu-t(2) country(16) us(840) organization(1) netscape(113730)} */ - -/* ISO arc for standard certificate and CRL extensions */ -#define MBEDTLS_OID_ID_CE MBEDTLS_OID_ISO_CCITT_DS "\x1D" /**< id-ce OBJECT IDENTIFIER ::= {joint-iso-ccitt(2) ds(5) 29} */ - -#define MBEDTLS_OID_NIST_ALG MBEDTLS_OID_GOV "\x03\x04" /** { joint-iso-itu-t(2) country(16) us(840) organization(1) gov(101) csor(3) nistAlgorithm(4) */ - -/** - * Private Internet Extensions - * { iso(1) identified-organization(3) dod(6) internet(1) - * security(5) mechanisms(5) pkix(7) } - */ -#define MBEDTLS_OID_INTERNET MBEDTLS_OID_ISO_IDENTIFIED_ORG MBEDTLS_OID_ORG_DOD "\x01" -#define MBEDTLS_OID_PKIX MBEDTLS_OID_INTERNET "\x05\x05\x07" - -/* - * Arc for standard naming attributes - */ -#define MBEDTLS_OID_AT MBEDTLS_OID_ISO_CCITT_DS "\x04" /**< id-at OBJECT IDENTIFIER ::= {joint-iso-ccitt(2) ds(5) 4} */ -#define MBEDTLS_OID_AT_CN MBEDTLS_OID_AT "\x03" /**< id-at-commonName AttributeType:= {id-at 3} */ -#define MBEDTLS_OID_AT_SUR_NAME MBEDTLS_OID_AT "\x04" /**< id-at-surName AttributeType:= {id-at 4} */ -#define MBEDTLS_OID_AT_SERIAL_NUMBER MBEDTLS_OID_AT "\x05" /**< id-at-serialNumber AttributeType:= {id-at 5} */ -#define MBEDTLS_OID_AT_COUNTRY MBEDTLS_OID_AT "\x06" /**< id-at-countryName AttributeType:= {id-at 6} */ -#define MBEDTLS_OID_AT_LOCALITY MBEDTLS_OID_AT "\x07" /**< id-at-locality AttributeType:= {id-at 7} */ -#define MBEDTLS_OID_AT_STATE MBEDTLS_OID_AT "\x08" /**< id-at-state AttributeType:= {id-at 8} */ -#define MBEDTLS_OID_AT_ORGANIZATION MBEDTLS_OID_AT "\x0A" /**< id-at-organizationName AttributeType:= {id-at 10} */ -#define MBEDTLS_OID_AT_ORG_UNIT MBEDTLS_OID_AT "\x0B" /**< id-at-organizationalUnitName AttributeType:= {id-at 11} */ -#define MBEDTLS_OID_AT_TITLE MBEDTLS_OID_AT "\x0C" /**< id-at-title AttributeType:= {id-at 12} */ -#define MBEDTLS_OID_AT_POSTAL_ADDRESS MBEDTLS_OID_AT "\x10" /**< id-at-postalAddress AttributeType:= {id-at 16} */ -#define MBEDTLS_OID_AT_POSTAL_CODE MBEDTLS_OID_AT "\x11" /**< id-at-postalCode AttributeType:= {id-at 17} */ -#define MBEDTLS_OID_AT_GIVEN_NAME MBEDTLS_OID_AT "\x2A" /**< id-at-givenName AttributeType:= {id-at 42} */ -#define MBEDTLS_OID_AT_INITIALS MBEDTLS_OID_AT "\x2B" /**< id-at-initials AttributeType:= {id-at 43} */ -#define MBEDTLS_OID_AT_GENERATION_QUALIFIER MBEDTLS_OID_AT "\x2C" /**< id-at-generationQualifier AttributeType:= {id-at 44} */ -#define MBEDTLS_OID_AT_UNIQUE_IDENTIFIER MBEDTLS_OID_AT "\x2D" /**< id-at-uniqueIdentifier AttributType:= {id-at 45} */ -#define MBEDTLS_OID_AT_DN_QUALIFIER MBEDTLS_OID_AT "\x2E" /**< id-at-dnQualifier AttributeType:= {id-at 46} */ -#define MBEDTLS_OID_AT_PSEUDONYM MBEDTLS_OID_AT "\x41" /**< id-at-pseudonym AttributeType:= {id-at 65} */ - -#define MBEDTLS_OID_UID "\x09\x92\x26\x89\x93\xF2\x2C\x64\x01\x01" /** id-domainComponent AttributeType:= {itu-t(0) data(9) pss(2342) ucl(19200300) pilot(100) pilotAttributeType(1) uid(1)} */ -#define MBEDTLS_OID_DOMAIN_COMPONENT "\x09\x92\x26\x89\x93\xF2\x2C\x64\x01\x19" /** id-domainComponent AttributeType:= {itu-t(0) data(9) pss(2342) ucl(19200300) pilot(100) pilotAttributeType(1) domainComponent(25)} */ - -/* - * OIDs for standard certificate extensions - */ -#define MBEDTLS_OID_AUTHORITY_KEY_IDENTIFIER MBEDTLS_OID_ID_CE "\x23" /**< id-ce-authorityKeyIdentifier OBJECT IDENTIFIER ::= { id-ce 35 } */ -#define MBEDTLS_OID_SUBJECT_KEY_IDENTIFIER MBEDTLS_OID_ID_CE "\x0E" /**< id-ce-subjectKeyIdentifier OBJECT IDENTIFIER ::= { id-ce 14 } */ -#define MBEDTLS_OID_KEY_USAGE MBEDTLS_OID_ID_CE "\x0F" /**< id-ce-keyUsage OBJECT IDENTIFIER ::= { id-ce 15 } */ -#define MBEDTLS_OID_CERTIFICATE_POLICIES MBEDTLS_OID_ID_CE "\x20" /**< id-ce-certificatePolicies OBJECT IDENTIFIER ::= { id-ce 32 } */ -#define MBEDTLS_OID_POLICY_MAPPINGS MBEDTLS_OID_ID_CE "\x21" /**< id-ce-policyMappings OBJECT IDENTIFIER ::= { id-ce 33 } */ -#define MBEDTLS_OID_SUBJECT_ALT_NAME MBEDTLS_OID_ID_CE "\x11" /**< id-ce-subjectAltName OBJECT IDENTIFIER ::= { id-ce 17 } */ -#define MBEDTLS_OID_ISSUER_ALT_NAME MBEDTLS_OID_ID_CE "\x12" /**< id-ce-issuerAltName OBJECT IDENTIFIER ::= { id-ce 18 } */ -#define MBEDTLS_OID_SUBJECT_DIRECTORY_ATTRS MBEDTLS_OID_ID_CE "\x09" /**< id-ce-subjectDirectoryAttributes OBJECT IDENTIFIER ::= { id-ce 9 } */ -#define MBEDTLS_OID_BASIC_CONSTRAINTS MBEDTLS_OID_ID_CE "\x13" /**< id-ce-basicConstraints OBJECT IDENTIFIER ::= { id-ce 19 } */ -#define MBEDTLS_OID_NAME_CONSTRAINTS MBEDTLS_OID_ID_CE "\x1E" /**< id-ce-nameConstraints OBJECT IDENTIFIER ::= { id-ce 30 } */ -#define MBEDTLS_OID_POLICY_CONSTRAINTS MBEDTLS_OID_ID_CE "\x24" /**< id-ce-policyConstraints OBJECT IDENTIFIER ::= { id-ce 36 } */ -#define MBEDTLS_OID_EXTENDED_KEY_USAGE MBEDTLS_OID_ID_CE "\x25" /**< id-ce-extKeyUsage OBJECT IDENTIFIER ::= { id-ce 37 } */ -#define MBEDTLS_OID_CRL_DISTRIBUTION_POINTS MBEDTLS_OID_ID_CE "\x1F" /**< id-ce-cRLDistributionPoints OBJECT IDENTIFIER ::= { id-ce 31 } */ -#define MBEDTLS_OID_INIHIBIT_ANYPOLICY MBEDTLS_OID_ID_CE "\x36" /**< id-ce-inhibitAnyPolicy OBJECT IDENTIFIER ::= { id-ce 54 } */ -#define MBEDTLS_OID_FRESHEST_CRL MBEDTLS_OID_ID_CE "\x2E" /**< id-ce-freshestCRL OBJECT IDENTIFIER ::= { id-ce 46 } */ - -/* - * Certificate policies - */ -#define MBEDTLS_OID_ANY_POLICY MBEDTLS_OID_CERTIFICATE_POLICIES "\x00" /**< anyPolicy OBJECT IDENTIFIER ::= { id-ce-certificatePolicies 0 } */ - -/* - * Netscape certificate extensions - */ -#define MBEDTLS_OID_NS_CERT MBEDTLS_OID_NETSCAPE "\x01" -#define MBEDTLS_OID_NS_CERT_TYPE MBEDTLS_OID_NS_CERT "\x01" -#define MBEDTLS_OID_NS_BASE_URL MBEDTLS_OID_NS_CERT "\x02" -#define MBEDTLS_OID_NS_REVOCATION_URL MBEDTLS_OID_NS_CERT "\x03" -#define MBEDTLS_OID_NS_CA_REVOCATION_URL MBEDTLS_OID_NS_CERT "\x04" -#define MBEDTLS_OID_NS_RENEWAL_URL MBEDTLS_OID_NS_CERT "\x07" -#define MBEDTLS_OID_NS_CA_POLICY_URL MBEDTLS_OID_NS_CERT "\x08" -#define MBEDTLS_OID_NS_SSL_SERVER_NAME MBEDTLS_OID_NS_CERT "\x0C" -#define MBEDTLS_OID_NS_COMMENT MBEDTLS_OID_NS_CERT "\x0D" -#define MBEDTLS_OID_NS_DATA_TYPE MBEDTLS_OID_NETSCAPE "\x02" -#define MBEDTLS_OID_NS_CERT_SEQUENCE MBEDTLS_OID_NS_DATA_TYPE "\x05" - -/* - * OIDs for CRL extensions - */ -#define MBEDTLS_OID_PRIVATE_KEY_USAGE_PERIOD MBEDTLS_OID_ID_CE "\x10" -#define MBEDTLS_OID_CRL_NUMBER MBEDTLS_OID_ID_CE "\x14" /**< id-ce-cRLNumber OBJECT IDENTIFIER ::= { id-ce 20 } */ - -/* - * X.509 v3 Extended key usage OIDs - */ -#define MBEDTLS_OID_ANY_EXTENDED_KEY_USAGE MBEDTLS_OID_EXTENDED_KEY_USAGE "\x00" /**< anyExtendedKeyUsage OBJECT IDENTIFIER ::= { id-ce-extKeyUsage 0 } */ - -#define MBEDTLS_OID_KP MBEDTLS_OID_PKIX "\x03" /**< id-kp OBJECT IDENTIFIER ::= { id-pkix 3 } */ -#define MBEDTLS_OID_SERVER_AUTH MBEDTLS_OID_KP "\x01" /**< id-kp-serverAuth OBJECT IDENTIFIER ::= { id-kp 1 } */ -#define MBEDTLS_OID_CLIENT_AUTH MBEDTLS_OID_KP "\x02" /**< id-kp-clientAuth OBJECT IDENTIFIER ::= { id-kp 2 } */ -#define MBEDTLS_OID_CODE_SIGNING MBEDTLS_OID_KP "\x03" /**< id-kp-codeSigning OBJECT IDENTIFIER ::= { id-kp 3 } */ -#define MBEDTLS_OID_EMAIL_PROTECTION MBEDTLS_OID_KP "\x04" /**< id-kp-emailProtection OBJECT IDENTIFIER ::= { id-kp 4 } */ -#define MBEDTLS_OID_TIME_STAMPING MBEDTLS_OID_KP "\x08" /**< id-kp-timeStamping OBJECT IDENTIFIER ::= { id-kp 8 } */ -#define MBEDTLS_OID_OCSP_SIGNING MBEDTLS_OID_KP "\x09" /**< id-kp-OCSPSigning OBJECT IDENTIFIER ::= { id-kp 9 } */ - -/** - * Wi-SUN Alliance Field Area Network - * { iso(1) identified-organization(3) dod(6) internet(1) - * private(4) enterprise(1) WiSUN(45605) FieldAreaNetwork(1) } - */ -#define MBEDTLS_OID_WISUN_FAN MBEDTLS_OID_INTERNET "\x04\x01\x82\xe4\x25\x01" - -#define MBEDTLS_OID_ON MBEDTLS_OID_PKIX "\x08" /**< id-on OBJECT IDENTIFIER ::= { id-pkix 8 } */ -#define MBEDTLS_OID_ON_HW_MODULE_NAME MBEDTLS_OID_ON "\x04" /**< id-on-hardwareModuleName OBJECT IDENTIFIER ::= { id-on 4 } */ - -/* - * PKCS definition OIDs - */ - -#define MBEDTLS_OID_PKCS MBEDTLS_OID_RSA_COMPANY "\x01" /**< pkcs OBJECT IDENTIFIER ::= { iso(1) member-body(2) us(840) rsadsi(113549) 1 } */ -#define MBEDTLS_OID_PKCS1 MBEDTLS_OID_PKCS "\x01" /**< pkcs-1 OBJECT IDENTIFIER ::= { iso(1) member-body(2) us(840) rsadsi(113549) pkcs(1) 1 } */ -#define MBEDTLS_OID_PKCS5 MBEDTLS_OID_PKCS "\x05" /**< pkcs-5 OBJECT IDENTIFIER ::= { iso(1) member-body(2) us(840) rsadsi(113549) pkcs(1) 5 } */ -#define MBEDTLS_OID_PKCS9 MBEDTLS_OID_PKCS "\x09" /**< pkcs-9 OBJECT IDENTIFIER ::= { iso(1) member-body(2) us(840) rsadsi(113549) pkcs(1) 9 } */ -#define MBEDTLS_OID_PKCS12 MBEDTLS_OID_PKCS "\x0c" /**< pkcs-12 OBJECT IDENTIFIER ::= { iso(1) member-body(2) us(840) rsadsi(113549) pkcs(1) 12 } */ - -/* - * PKCS#1 OIDs - */ -#define MBEDTLS_OID_PKCS1_RSA MBEDTLS_OID_PKCS1 "\x01" /**< rsaEncryption OBJECT IDENTIFIER ::= { pkcs-1 1 } */ -#define MBEDTLS_OID_PKCS1_MD5 MBEDTLS_OID_PKCS1 "\x04" /**< md5WithRSAEncryption ::= { pkcs-1 4 } */ -#define MBEDTLS_OID_PKCS1_SHA1 MBEDTLS_OID_PKCS1 "\x05" /**< sha1WithRSAEncryption ::= { pkcs-1 5 } */ -#define MBEDTLS_OID_PKCS1_SHA224 MBEDTLS_OID_PKCS1 "\x0e" /**< sha224WithRSAEncryption ::= { pkcs-1 14 } */ -#define MBEDTLS_OID_PKCS1_SHA256 MBEDTLS_OID_PKCS1 "\x0b" /**< sha256WithRSAEncryption ::= { pkcs-1 11 } */ -#define MBEDTLS_OID_PKCS1_SHA384 MBEDTLS_OID_PKCS1 "\x0c" /**< sha384WithRSAEncryption ::= { pkcs-1 12 } */ -#define MBEDTLS_OID_PKCS1_SHA512 MBEDTLS_OID_PKCS1 "\x0d" /**< sha512WithRSAEncryption ::= { pkcs-1 13 } */ - -#define MBEDTLS_OID_RSA_SHA_OBS "\x2B\x0E\x03\x02\x1D" - -#define MBEDTLS_OID_PKCS9_EMAIL MBEDTLS_OID_PKCS9 "\x01" /**< emailAddress AttributeType ::= { pkcs-9 1 } */ - -/* RFC 4055 */ -#define MBEDTLS_OID_RSASSA_PSS MBEDTLS_OID_PKCS1 "\x0a" /**< id-RSASSA-PSS ::= { pkcs-1 10 } */ -#define MBEDTLS_OID_MGF1 MBEDTLS_OID_PKCS1 "\x08" /**< id-mgf1 ::= { pkcs-1 8 } */ - -/* - * Digest algorithms - */ -#define MBEDTLS_OID_DIGEST_ALG_MD5 MBEDTLS_OID_RSA_COMPANY "\x02\x05" /**< id-mbedtls_md5 OBJECT IDENTIFIER ::= { iso(1) member-body(2) us(840) rsadsi(113549) digestAlgorithm(2) 5 } */ -#define MBEDTLS_OID_DIGEST_ALG_SHA1 MBEDTLS_OID_ISO_IDENTIFIED_ORG MBEDTLS_OID_OIW_SECSIG_SHA1 /**< id-mbedtls_sha1 OBJECT IDENTIFIER ::= { iso(1) identified-organization(3) oiw(14) secsig(3) algorithms(2) 26 } */ -#define MBEDTLS_OID_DIGEST_ALG_SHA224 MBEDTLS_OID_NIST_ALG "\x02\x04" /**< id-sha224 OBJECT IDENTIFIER ::= { joint-iso-itu-t(2) country(16) us(840) organization(1) gov(101) csor(3) nistalgorithm(4) hashalgs(2) 4 } */ -#define MBEDTLS_OID_DIGEST_ALG_SHA256 MBEDTLS_OID_NIST_ALG "\x02\x01" /**< id-mbedtls_sha256 OBJECT IDENTIFIER ::= { joint-iso-itu-t(2) country(16) us(840) organization(1) gov(101) csor(3) nistalgorithm(4) hashalgs(2) 1 } */ - -#define MBEDTLS_OID_DIGEST_ALG_SHA384 MBEDTLS_OID_NIST_ALG "\x02\x02" /**< id-sha384 OBJECT IDENTIFIER ::= { joint-iso-itu-t(2) country(16) us(840) organization(1) gov(101) csor(3) nistalgorithm(4) hashalgs(2) 2 } */ - -#define MBEDTLS_OID_DIGEST_ALG_SHA512 MBEDTLS_OID_NIST_ALG "\x02\x03" /**< id-mbedtls_sha512 OBJECT IDENTIFIER ::= { joint-iso-itu-t(2) country(16) us(840) organization(1) gov(101) csor(3) nistalgorithm(4) hashalgs(2) 3 } */ - -#define MBEDTLS_OID_DIGEST_ALG_RIPEMD160 MBEDTLS_OID_TELETRUST "\x03\x02\x01" /**< id-ripemd160 OBJECT IDENTIFIER :: { iso(1) identified-organization(3) teletrust(36) algorithm(3) hashAlgorithm(2) ripemd160(1) } */ - -#define MBEDTLS_OID_HMAC_SHA1 MBEDTLS_OID_RSA_COMPANY "\x02\x07" /**< id-hmacWithSHA1 OBJECT IDENTIFIER ::= { iso(1) member-body(2) us(840) rsadsi(113549) digestAlgorithm(2) 7 } */ - -#define MBEDTLS_OID_HMAC_SHA224 MBEDTLS_OID_RSA_COMPANY "\x02\x08" /**< id-hmacWithSHA224 OBJECT IDENTIFIER ::= { iso(1) member-body(2) us(840) rsadsi(113549) digestAlgorithm(2) 8 } */ - -#define MBEDTLS_OID_HMAC_SHA256 MBEDTLS_OID_RSA_COMPANY "\x02\x09" /**< id-hmacWithSHA256 OBJECT IDENTIFIER ::= { iso(1) member-body(2) us(840) rsadsi(113549) digestAlgorithm(2) 9 } */ - -#define MBEDTLS_OID_HMAC_SHA384 MBEDTLS_OID_RSA_COMPANY "\x02\x0A" /**< id-hmacWithSHA384 OBJECT IDENTIFIER ::= { iso(1) member-body(2) us(840) rsadsi(113549) digestAlgorithm(2) 10 } */ - -#define MBEDTLS_OID_HMAC_SHA512 MBEDTLS_OID_RSA_COMPANY "\x02\x0B" /**< id-hmacWithSHA512 OBJECT IDENTIFIER ::= { iso(1) member-body(2) us(840) rsadsi(113549) digestAlgorithm(2) 11 } */ - -/* - * Encryption algorithms - */ -#define MBEDTLS_OID_DES_CBC MBEDTLS_OID_ISO_IDENTIFIED_ORG MBEDTLS_OID_OIW_SECSIG_ALG "\x07" /**< desCBC OBJECT IDENTIFIER ::= { iso(1) identified-organization(3) oiw(14) secsig(3) algorithms(2) 7 } */ -#define MBEDTLS_OID_DES_EDE3_CBC MBEDTLS_OID_RSA_COMPANY "\x03\x07" /**< des-ede3-cbc OBJECT IDENTIFIER ::= { iso(1) member-body(2) -- us(840) rsadsi(113549) encryptionAlgorithm(3) 7 } */ -#define MBEDTLS_OID_AES MBEDTLS_OID_NIST_ALG "\x01" /** aes OBJECT IDENTIFIER ::= { joint-iso-itu-t(2) country(16) us(840) organization(1) gov(101) csor(3) nistAlgorithm(4) 1 } */ - -/* - * Key Wrapping algorithms - */ -/* - * RFC 5649 - */ -#define MBEDTLS_OID_AES128_KW MBEDTLS_OID_AES "\x05" /** id-aes128-wrap OBJECT IDENTIFIER ::= { aes 5 } */ -#define MBEDTLS_OID_AES128_KWP MBEDTLS_OID_AES "\x08" /** id-aes128-wrap-pad OBJECT IDENTIFIER ::= { aes 8 } */ -#define MBEDTLS_OID_AES192_KW MBEDTLS_OID_AES "\x19" /** id-aes192-wrap OBJECT IDENTIFIER ::= { aes 25 } */ -#define MBEDTLS_OID_AES192_KWP MBEDTLS_OID_AES "\x1c" /** id-aes192-wrap-pad OBJECT IDENTIFIER ::= { aes 28 } */ -#define MBEDTLS_OID_AES256_KW MBEDTLS_OID_AES "\x2d" /** id-aes256-wrap OBJECT IDENTIFIER ::= { aes 45 } */ -#define MBEDTLS_OID_AES256_KWP MBEDTLS_OID_AES "\x30" /** id-aes256-wrap-pad OBJECT IDENTIFIER ::= { aes 48 } */ -/* - * PKCS#5 OIDs - */ -#define MBEDTLS_OID_PKCS5_PBKDF2 MBEDTLS_OID_PKCS5 "\x0c" /**< id-PBKDF2 OBJECT IDENTIFIER ::= {pkcs-5 12} */ -#define MBEDTLS_OID_PKCS5_PBES2 MBEDTLS_OID_PKCS5 "\x0d" /**< id-PBES2 OBJECT IDENTIFIER ::= {pkcs-5 13} */ -#define MBEDTLS_OID_PKCS5_PBMAC1 MBEDTLS_OID_PKCS5 "\x0e" /**< id-PBMAC1 OBJECT IDENTIFIER ::= {pkcs-5 14} */ - -/* - * PKCS#5 PBES1 algorithms - */ -#define MBEDTLS_OID_PKCS5_PBE_MD5_DES_CBC MBEDTLS_OID_PKCS5 "\x03" /**< pbeWithMD5AndDES-CBC OBJECT IDENTIFIER ::= {pkcs-5 3} */ -#define MBEDTLS_OID_PKCS5_PBE_MD5_RC2_CBC MBEDTLS_OID_PKCS5 "\x06" /**< pbeWithMD5AndRC2-CBC OBJECT IDENTIFIER ::= {pkcs-5 6} */ -#define MBEDTLS_OID_PKCS5_PBE_SHA1_DES_CBC MBEDTLS_OID_PKCS5 "\x0a" /**< pbeWithSHA1AndDES-CBC OBJECT IDENTIFIER ::= {pkcs-5 10} */ -#define MBEDTLS_OID_PKCS5_PBE_SHA1_RC2_CBC MBEDTLS_OID_PKCS5 "\x0b" /**< pbeWithSHA1AndRC2-CBC OBJECT IDENTIFIER ::= {pkcs-5 11} */ - -/* - * PKCS#8 OIDs - */ -#define MBEDTLS_OID_PKCS9_CSR_EXT_REQ MBEDTLS_OID_PKCS9 "\x0e" /**< extensionRequest OBJECT IDENTIFIER ::= {pkcs-9 14} */ - -/* - * PKCS#12 PBE OIDs - */ -#define MBEDTLS_OID_PKCS12_PBE MBEDTLS_OID_PKCS12 "\x01" /**< pkcs-12PbeIds OBJECT IDENTIFIER ::= {pkcs-12 1} */ - -#define MBEDTLS_OID_PKCS12_PBE_SHA1_DES3_EDE_CBC MBEDTLS_OID_PKCS12_PBE "\x03" /**< pbeWithSHAAnd3-KeyTripleDES-CBC OBJECT IDENTIFIER ::= {pkcs-12PbeIds 3} */ -#define MBEDTLS_OID_PKCS12_PBE_SHA1_DES2_EDE_CBC MBEDTLS_OID_PKCS12_PBE "\x04" /**< pbeWithSHAAnd2-KeyTripleDES-CBC OBJECT IDENTIFIER ::= {pkcs-12PbeIds 4} */ -#define MBEDTLS_OID_PKCS12_PBE_SHA1_RC2_128_CBC MBEDTLS_OID_PKCS12_PBE "\x05" /**< pbeWithSHAAnd128BitRC2-CBC OBJECT IDENTIFIER ::= {pkcs-12PbeIds 5} */ -#define MBEDTLS_OID_PKCS12_PBE_SHA1_RC2_40_CBC MBEDTLS_OID_PKCS12_PBE "\x06" /**< pbeWithSHAAnd40BitRC2-CBC OBJECT IDENTIFIER ::= {pkcs-12PbeIds 6} */ - -/* - * EC key algorithms from RFC 5480 - */ - -/* id-ecPublicKey OBJECT IDENTIFIER ::= { - * iso(1) member-body(2) us(840) ansi-X9-62(10045) keyType(2) 1 } */ -#define MBEDTLS_OID_EC_ALG_UNRESTRICTED MBEDTLS_OID_ANSI_X9_62 "\x02\01" - -/* id-ecDH OBJECT IDENTIFIER ::= { - * iso(1) identified-organization(3) certicom(132) - * schemes(1) ecdh(12) } */ -#define MBEDTLS_OID_EC_ALG_ECDH MBEDTLS_OID_CERTICOM "\x01\x0c" - -/* - * ECParameters namedCurve identifiers, from RFC 5480, RFC 5639, and SEC2 - */ - -/* secp192r1 OBJECT IDENTIFIER ::= { - * iso(1) member-body(2) us(840) ansi-X9-62(10045) curves(3) prime(1) 1 } */ -#define MBEDTLS_OID_EC_GRP_SECP192R1 MBEDTLS_OID_ANSI_X9_62 "\x03\x01\x01" - -/* secp224r1 OBJECT IDENTIFIER ::= { - * iso(1) identified-organization(3) certicom(132) curve(0) 33 } */ -#define MBEDTLS_OID_EC_GRP_SECP224R1 MBEDTLS_OID_CERTICOM "\x00\x21" - -/* secp256r1 OBJECT IDENTIFIER ::= { - * iso(1) member-body(2) us(840) ansi-X9-62(10045) curves(3) prime(1) 7 } */ -#define MBEDTLS_OID_EC_GRP_SECP256R1 MBEDTLS_OID_ANSI_X9_62 "\x03\x01\x07" - -/* secp384r1 OBJECT IDENTIFIER ::= { - * iso(1) identified-organization(3) certicom(132) curve(0) 34 } */ -#define MBEDTLS_OID_EC_GRP_SECP384R1 MBEDTLS_OID_CERTICOM "\x00\x22" - -/* secp521r1 OBJECT IDENTIFIER ::= { - * iso(1) identified-organization(3) certicom(132) curve(0) 35 } */ -#define MBEDTLS_OID_EC_GRP_SECP521R1 MBEDTLS_OID_CERTICOM "\x00\x23" - -/* secp192k1 OBJECT IDENTIFIER ::= { - * iso(1) identified-organization(3) certicom(132) curve(0) 31 } */ -#define MBEDTLS_OID_EC_GRP_SECP192K1 MBEDTLS_OID_CERTICOM "\x00\x1f" - -/* secp224k1 OBJECT IDENTIFIER ::= { - * iso(1) identified-organization(3) certicom(132) curve(0) 32 } */ -#define MBEDTLS_OID_EC_GRP_SECP224K1 MBEDTLS_OID_CERTICOM "\x00\x20" - -/* secp256k1 OBJECT IDENTIFIER ::= { - * iso(1) identified-organization(3) certicom(132) curve(0) 10 } */ -#define MBEDTLS_OID_EC_GRP_SECP256K1 MBEDTLS_OID_CERTICOM "\x00\x0a" - -/* RFC 5639 4.1 - * ecStdCurvesAndGeneration OBJECT IDENTIFIER::= {iso(1) - * identified-organization(3) teletrust(36) algorithm(3) signature- - * algorithm(3) ecSign(2) 8} - * ellipticCurve OBJECT IDENTIFIER ::= {ecStdCurvesAndGeneration 1} - * versionOne OBJECT IDENTIFIER ::= {ellipticCurve 1} */ -#define MBEDTLS_OID_EC_BRAINPOOL_V1 MBEDTLS_OID_TELETRUST "\x03\x03\x02\x08\x01\x01" - -/* brainpoolP256r1 OBJECT IDENTIFIER ::= {versionOne 7} */ -#define MBEDTLS_OID_EC_GRP_BP256R1 MBEDTLS_OID_EC_BRAINPOOL_V1 "\x07" - -/* brainpoolP384r1 OBJECT IDENTIFIER ::= {versionOne 11} */ -#define MBEDTLS_OID_EC_GRP_BP384R1 MBEDTLS_OID_EC_BRAINPOOL_V1 "\x0B" - -/* brainpoolP512r1 OBJECT IDENTIFIER ::= {versionOne 13} */ -#define MBEDTLS_OID_EC_GRP_BP512R1 MBEDTLS_OID_EC_BRAINPOOL_V1 "\x0D" - -/* - * SEC1 C.1 - * - * prime-field OBJECT IDENTIFIER ::= { id-fieldType 1 } - * id-fieldType OBJECT IDENTIFIER ::= { ansi-X9-62 fieldType(1)} - */ -#define MBEDTLS_OID_ANSI_X9_62_FIELD_TYPE MBEDTLS_OID_ANSI_X9_62 "\x01" -#define MBEDTLS_OID_ANSI_X9_62_PRIME_FIELD MBEDTLS_OID_ANSI_X9_62_FIELD_TYPE "\x01" - -/* - * ECDSA signature identifiers, from RFC 5480 - */ -#define MBEDTLS_OID_ANSI_X9_62_SIG MBEDTLS_OID_ANSI_X9_62 "\x04" /* signatures(4) */ -#define MBEDTLS_OID_ANSI_X9_62_SIG_SHA2 MBEDTLS_OID_ANSI_X9_62_SIG "\x03" /* ecdsa-with-SHA2(3) */ - -/* ecdsa-with-SHA1 OBJECT IDENTIFIER ::= { - * iso(1) member-body(2) us(840) ansi-X9-62(10045) signatures(4) 1 } */ -#define MBEDTLS_OID_ECDSA_SHA1 MBEDTLS_OID_ANSI_X9_62_SIG "\x01" - -/* ecdsa-with-SHA224 OBJECT IDENTIFIER ::= { - * iso(1) member-body(2) us(840) ansi-X9-62(10045) signatures(4) - * ecdsa-with-SHA2(3) 1 } */ -#define MBEDTLS_OID_ECDSA_SHA224 MBEDTLS_OID_ANSI_X9_62_SIG_SHA2 "\x01" - -/* ecdsa-with-SHA256 OBJECT IDENTIFIER ::= { - * iso(1) member-body(2) us(840) ansi-X9-62(10045) signatures(4) - * ecdsa-with-SHA2(3) 2 } */ -#define MBEDTLS_OID_ECDSA_SHA256 MBEDTLS_OID_ANSI_X9_62_SIG_SHA2 "\x02" - -/* ecdsa-with-SHA384 OBJECT IDENTIFIER ::= { - * iso(1) member-body(2) us(840) ansi-X9-62(10045) signatures(4) - * ecdsa-with-SHA2(3) 3 } */ -#define MBEDTLS_OID_ECDSA_SHA384 MBEDTLS_OID_ANSI_X9_62_SIG_SHA2 "\x03" - -/* ecdsa-with-SHA512 OBJECT IDENTIFIER ::= { - * iso(1) member-body(2) us(840) ansi-X9-62(10045) signatures(4) - * ecdsa-with-SHA2(3) 4 } */ -#define MBEDTLS_OID_ECDSA_SHA512 MBEDTLS_OID_ANSI_X9_62_SIG_SHA2 "\x04" - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief Base OID descriptor structure - */ -typedef struct mbedtls_oid_descriptor_t -{ - const char *MBEDTLS_PRIVATE(asn1); /*!< OID ASN.1 representation */ - size_t MBEDTLS_PRIVATE(asn1_len); /*!< length of asn1 */ -#if !defined(MBEDTLS_X509_REMOVE_INFO) - const char *MBEDTLS_PRIVATE(name); /*!< official name (e.g. from RFC) */ - const char *MBEDTLS_PRIVATE(description); /*!< human friendly description */ -#endif -} mbedtls_oid_descriptor_t; - -/** - * \brief Translate an ASN.1 OID into its numeric representation - * (e.g. "\x2A\x86\x48\x86\xF7\x0D" into "1.2.840.113549") - * - * \param buf buffer to put representation in - * \param size size of the buffer - * \param oid OID to translate - * - * \return Length of the string written (excluding final NULL) or - * MBEDTLS_ERR_OID_BUF_TOO_SMALL in case of error - */ -int mbedtls_oid_get_numeric_string( char *buf, size_t size, const mbedtls_asn1_buf *oid ); - -/** - * \brief Translate an X.509 extension OID into local values - * - * \param oid OID to use - * \param ext_type place to store the extension type - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_x509_ext_type( const mbedtls_asn1_buf *oid, int *ext_type ); - -/** - * \brief Translate an X.509 attribute type OID into the short name - * (e.g. the OID for an X520 Common Name into "CN") - * - * \param oid OID to use - * \param short_name place to store the string pointer - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_attr_short_name( const mbedtls_asn1_buf *oid, const char **short_name ); - -/** - * \brief Translate PublicKeyAlgorithm OID into pk_type - * - * \param oid OID to use - * \param pk_alg place to store public key algorithm - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_pk_alg( const mbedtls_asn1_buf *oid, mbedtls_pk_type_t *pk_alg ); - -/** - * \brief Translate pk_type into PublicKeyAlgorithm OID - * - * \param pk_alg Public key type to look for - * \param oid place to store ASN.1 OID string pointer - * \param olen length of the OID - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_oid_by_pk_alg( mbedtls_pk_type_t pk_alg, - const char **oid, size_t *olen ); - -#if defined(MBEDTLS_ECP_C) -/** - * \brief Translate NamedCurve OID into an EC group identifier - * - * \param oid OID to use - * \param grp_id place to store group id - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_ec_grp( const mbedtls_asn1_buf *oid, mbedtls_ecp_group_id *grp_id ); - -/** - * \brief Translate EC group identifier into NamedCurve OID - * - * \param grp_id EC group identifier - * \param oid place to store ASN.1 OID string pointer - * \param olen length of the OID - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_oid_by_ec_grp( mbedtls_ecp_group_id grp_id, - const char **oid, size_t *olen ); -#endif /* MBEDTLS_ECP_C */ - -#if defined(MBEDTLS_MD_C) -/** - * \brief Translate SignatureAlgorithm OID into md_type and pk_type - * - * \param oid OID to use - * \param md_alg place to store message digest algorithm - * \param pk_alg place to store public key algorithm - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_sig_alg( const mbedtls_asn1_buf *oid, - mbedtls_md_type_t *md_alg, mbedtls_pk_type_t *pk_alg ); - -/** - * \brief Translate SignatureAlgorithm OID into description - * - * \param oid OID to use - * \param desc place to store string pointer - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_sig_alg_desc( const mbedtls_asn1_buf *oid, const char **desc ); - -/** - * \brief Translate md_type and pk_type into SignatureAlgorithm OID - * - * \param md_alg message digest algorithm - * \param pk_alg public key algorithm - * \param oid place to store ASN.1 OID string pointer - * \param olen length of the OID - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_oid_by_sig_alg( mbedtls_pk_type_t pk_alg, mbedtls_md_type_t md_alg, - const char **oid, size_t *olen ); - -/** - * \brief Translate hash algorithm OID into md_type - * - * \param oid OID to use - * \param md_alg place to store message digest algorithm - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_md_alg( const mbedtls_asn1_buf *oid, mbedtls_md_type_t *md_alg ); - -/** - * \brief Translate hmac algorithm OID into md_type - * - * \param oid OID to use - * \param md_hmac place to store message hmac algorithm - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_md_hmac( const mbedtls_asn1_buf *oid, mbedtls_md_type_t *md_hmac ); -#endif /* MBEDTLS_MD_C */ - -#if !defined(MBEDTLS_X509_REMOVE_INFO) -/** - * \brief Translate Extended Key Usage OID into description - * - * \param oid OID to use - * \param desc place to store string pointer - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_extended_key_usage( const mbedtls_asn1_buf *oid, const char **desc ); -#endif - -/** - * \brief Translate certificate policies OID into description - * - * \param oid OID to use - * \param desc place to store string pointer - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_certificate_policies( const mbedtls_asn1_buf *oid, const char **desc ); - -/** - * \brief Translate md_type into hash algorithm OID - * - * \param md_alg message digest algorithm - * \param oid place to store ASN.1 OID string pointer - * \param olen length of the OID - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_oid_by_md( mbedtls_md_type_t md_alg, const char **oid, size_t *olen ); - -#if defined(MBEDTLS_CIPHER_C) -/** - * \brief Translate encryption algorithm OID into cipher_type - * - * \param oid OID to use - * \param cipher_alg place to store cipher algorithm - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_cipher_alg( const mbedtls_asn1_buf *oid, mbedtls_cipher_type_t *cipher_alg ); -#endif /* MBEDTLS_CIPHER_C */ - -#if defined(MBEDTLS_PKCS12_C) -/** - * \brief Translate PKCS#12 PBE algorithm OID into md_type and - * cipher_type - * - * \param oid OID to use - * \param md_alg place to store message digest algorithm - * \param cipher_alg place to store cipher algorithm - * - * \return 0 if successful, or MBEDTLS_ERR_OID_NOT_FOUND - */ -int mbedtls_oid_get_pkcs12_pbe_alg( const mbedtls_asn1_buf *oid, mbedtls_md_type_t *md_alg, - mbedtls_cipher_type_t *cipher_alg ); -#endif /* MBEDTLS_PKCS12_C */ - -#ifdef __cplusplus -} -#endif - -#endif /* oid.h */ diff --git a/windows/mbedtls/pem.h b/windows/mbedtls/pem.h deleted file mode 100644 index a01a296b..00000000 --- a/windows/mbedtls/pem.h +++ /dev/null @@ -1,141 +0,0 @@ -/** - * \file pem.h - * - * \brief Privacy Enhanced Mail (PEM) decoding - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_PEM_H -#define MBEDTLS_PEM_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include - -/** - * \name PEM Error codes - * These error codes are returned in case of errors reading the - * PEM data. - * \{ - */ -#define MBEDTLS_ERR_PEM_NO_HEADER_FOOTER_PRESENT -0x1080 /**< No PEM header or footer found. */ -#define MBEDTLS_ERR_PEM_INVALID_DATA -0x1100 /**< PEM string is not as expected. */ -#define MBEDTLS_ERR_PEM_ALLOC_FAILED -0x1180 /**< Failed to allocate memory. */ -#define MBEDTLS_ERR_PEM_INVALID_ENC_IV -0x1200 /**< RSA IV is not in hex-format. */ -#define MBEDTLS_ERR_PEM_UNKNOWN_ENC_ALG -0x1280 /**< Unsupported key encryption algorithm. */ -#define MBEDTLS_ERR_PEM_PASSWORD_REQUIRED -0x1300 /**< Private key password can't be empty. */ -#define MBEDTLS_ERR_PEM_PASSWORD_MISMATCH -0x1380 /**< Given private key password does not allow for correct decryption. */ -#define MBEDTLS_ERR_PEM_FEATURE_UNAVAILABLE -0x1400 /**< Unavailable feature, e.g. hashing/encryption combination. */ -#define MBEDTLS_ERR_PEM_BAD_INPUT_DATA -0x1480 /**< Bad input parameters to function. */ -/* \} name */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if defined(MBEDTLS_PEM_PARSE_C) -/** - * \brief PEM context structure - */ -typedef struct mbedtls_pem_context -{ - unsigned char *MBEDTLS_PRIVATE(buf); /*!< buffer for decoded data */ - size_t MBEDTLS_PRIVATE(buflen); /*!< length of the buffer */ - unsigned char *MBEDTLS_PRIVATE(info); /*!< buffer for extra header information */ -} -mbedtls_pem_context; - -/** - * \brief PEM context setup - * - * \param ctx context to be initialized - */ -void mbedtls_pem_init( mbedtls_pem_context *ctx ); - -/** - * \brief Read a buffer for PEM information and store the resulting - * data into the specified context buffers. - * - * \param ctx context to use - * \param header header string to seek and expect - * \param footer footer string to seek and expect - * \param data source data to look in (must be nul-terminated) - * \param pwd password for decryption (can be NULL) - * \param pwdlen length of password - * \param use_len destination for total length used (set after header is - * correctly read, so unless you get - * MBEDTLS_ERR_PEM_BAD_INPUT_DATA or - * MBEDTLS_ERR_PEM_NO_HEADER_FOOTER_PRESENT, use_len is - * the length to skip) - * - * \note Attempts to check password correctness by verifying if - * the decrypted text starts with an ASN.1 sequence of - * appropriate length - * - * \return 0 on success, or a specific PEM error code - */ -int mbedtls_pem_read_buffer( mbedtls_pem_context *ctx, const char *header, const char *footer, - const unsigned char *data, - const unsigned char *pwd, - size_t pwdlen, size_t *use_len ); - -/** - * \brief PEM context memory freeing - * - * \param ctx context to be freed - */ -void mbedtls_pem_free( mbedtls_pem_context *ctx ); -#endif /* MBEDTLS_PEM_PARSE_C */ - -#if defined(MBEDTLS_PEM_WRITE_C) -/** - * \brief Write a buffer of PEM information from a DER encoded - * buffer. - * - * \param header The header string to write. - * \param footer The footer string to write. - * \param der_data The DER data to encode. - * \param der_len The length of the DER data \p der_data in Bytes. - * \param buf The buffer to write to. - * \param buf_len The length of the output buffer \p buf in Bytes. - * \param olen The address at which to store the total length written - * or required (if \p buf_len is not enough). - * - * \note You may pass \c NULL for \p buf and \c 0 for \p buf_len - * to request the length of the resulting PEM buffer in - * `*olen`. - * - * \note This function may be called with overlapping \p der_data - * and \p buf buffers. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_BASE64_BUFFER_TOO_SMALL if \p buf isn't large - * enough to hold the PEM buffer. In this case, `*olen` holds - * the required minimum size of \p buf. - * \return Another PEM or BASE64 error code on other kinds of failure. - */ -int mbedtls_pem_write_buffer( const char *header, const char *footer, - const unsigned char *der_data, size_t der_len, - unsigned char *buf, size_t buf_len, size_t *olen ); -#endif /* MBEDTLS_PEM_WRITE_C */ - -#ifdef __cplusplus -} -#endif - -#endif /* pem.h */ diff --git a/windows/mbedtls/pk.h b/windows/mbedtls/pk.h deleted file mode 100644 index f3553ccc..00000000 --- a/windows/mbedtls/pk.h +++ /dev/null @@ -1,890 +0,0 @@ -/** - * \file pk.h - * - * \brief Public Key abstraction layer - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_PK_H -#define MBEDTLS_PK_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/md.h" - -#if defined(MBEDTLS_RSA_C) -#include "mbedtls/rsa.h" -#endif - -#if defined(MBEDTLS_ECP_C) -#include "mbedtls/ecp.h" -#endif - -#if defined(MBEDTLS_ECDSA_C) -#include "mbedtls/ecdsa.h" -#endif - -#if defined(MBEDTLS_USE_PSA_CRYPTO) -#include "psa/crypto.h" -#endif - -#if ( defined(__ARMCC_VERSION) || defined(_MSC_VER) ) && \ - !defined(inline) && !defined(__cplusplus) -#define inline __inline -#endif - -#define MBEDTLS_ERR_PK_ALLOC_FAILED -0x3F80 /**< Memory allocation failed. */ -#define MBEDTLS_ERR_PK_TYPE_MISMATCH -0x3F00 /**< Type mismatch, eg attempt to encrypt with an ECDSA key */ -#define MBEDTLS_ERR_PK_BAD_INPUT_DATA -0x3E80 /**< Bad input parameters to function. */ -#define MBEDTLS_ERR_PK_FILE_IO_ERROR -0x3E00 /**< Read/write of file failed. */ -#define MBEDTLS_ERR_PK_KEY_INVALID_VERSION -0x3D80 /**< Unsupported key version */ -#define MBEDTLS_ERR_PK_KEY_INVALID_FORMAT -0x3D00 /**< Invalid key tag or value. */ -#define MBEDTLS_ERR_PK_UNKNOWN_PK_ALG -0x3C80 /**< Key algorithm is unsupported (only RSA and EC are supported). */ -#define MBEDTLS_ERR_PK_PASSWORD_REQUIRED -0x3C00 /**< Private key password can't be empty. */ -#define MBEDTLS_ERR_PK_PASSWORD_MISMATCH -0x3B80 /**< Given private key password does not allow for correct decryption. */ -#define MBEDTLS_ERR_PK_INVALID_PUBKEY -0x3B00 /**< The pubkey tag or value is invalid (only RSA and EC are supported). */ -#define MBEDTLS_ERR_PK_INVALID_ALG -0x3A80 /**< The algorithm tag or value is invalid. */ -#define MBEDTLS_ERR_PK_UNKNOWN_NAMED_CURVE -0x3A00 /**< Elliptic curve is unsupported (only NIST curves are supported). */ -#define MBEDTLS_ERR_PK_FEATURE_UNAVAILABLE -0x3980 /**< Unavailable feature, e.g. RSA disabled for RSA key. */ -#define MBEDTLS_ERR_PK_SIG_LEN_MISMATCH -0x3900 /**< The buffer contains a valid signature followed by more data. */ -#define MBEDTLS_ERR_PK_BUFFER_TOO_SMALL -0x3880 /**< The output buffer is too small. */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief Public key types - */ -typedef enum { - MBEDTLS_PK_NONE=0, - MBEDTLS_PK_RSA, - MBEDTLS_PK_ECKEY, - MBEDTLS_PK_ECKEY_DH, - MBEDTLS_PK_ECDSA, - MBEDTLS_PK_RSA_ALT, - MBEDTLS_PK_RSASSA_PSS, - MBEDTLS_PK_OPAQUE, -} mbedtls_pk_type_t; - -/** - * \brief Options for RSASSA-PSS signature verification. - * See \c mbedtls_rsa_rsassa_pss_verify_ext() - */ -typedef struct mbedtls_pk_rsassa_pss_options -{ - mbedtls_md_type_t MBEDTLS_PRIVATE(mgf1_hash_id); - int MBEDTLS_PRIVATE(expected_salt_len); - -} mbedtls_pk_rsassa_pss_options; - -/** - * \brief Maximum size of a signature made by mbedtls_pk_sign(). - */ -/* We need to set MBEDTLS_PK_SIGNATURE_MAX_SIZE to the maximum signature - * size among the supported signature types. Do it by starting at 0, - * then incrementally increasing to be large enough for each supported - * signature mechanism. - * - * The resulting value can be 0, for example if MBEDTLS_ECDH_C is enabled - * (which allows the pk module to be included) but neither MBEDTLS_ECDSA_C - * nor MBEDTLS_RSA_C nor any opaque signature mechanism (PSA or RSA_ALT). - */ -#define MBEDTLS_PK_SIGNATURE_MAX_SIZE 0 - -#if ( defined(MBEDTLS_RSA_C) || defined(MBEDTLS_PK_RSA_ALT_SUPPORT) ) && \ - MBEDTLS_MPI_MAX_SIZE > MBEDTLS_PK_SIGNATURE_MAX_SIZE -/* For RSA, the signature can be as large as the bignum module allows. - * For RSA_ALT, the signature size is not necessarily tied to what the - * bignum module can do, but in the absence of any specific setting, - * we use that (rsa_alt_sign_wrap in library/pk_wrap.h will check). */ -#undef MBEDTLS_PK_SIGNATURE_MAX_SIZE -#define MBEDTLS_PK_SIGNATURE_MAX_SIZE MBEDTLS_MPI_MAX_SIZE -#endif - -#if defined(MBEDTLS_ECDSA_C) && \ - MBEDTLS_ECDSA_MAX_LEN > MBEDTLS_PK_SIGNATURE_MAX_SIZE -/* For ECDSA, the ecdsa module exports a constant for the maximum - * signature size. */ -#undef MBEDTLS_PK_SIGNATURE_MAX_SIZE -#define MBEDTLS_PK_SIGNATURE_MAX_SIZE MBEDTLS_ECDSA_MAX_LEN -#endif - -#if defined(MBEDTLS_USE_PSA_CRYPTO) -#if PSA_SIGNATURE_MAX_SIZE > MBEDTLS_PK_SIGNATURE_MAX_SIZE -/* PSA_SIGNATURE_MAX_SIZE is the maximum size of a signature made - * through the PSA API in the PSA representation. */ -#undef MBEDTLS_PK_SIGNATURE_MAX_SIZE -#define MBEDTLS_PK_SIGNATURE_MAX_SIZE PSA_SIGNATURE_MAX_SIZE -#endif - -#if PSA_VENDOR_ECDSA_SIGNATURE_MAX_SIZE + 11 > MBEDTLS_PK_SIGNATURE_MAX_SIZE -/* The Mbed TLS representation is different for ECDSA signatures: - * PSA uses the raw concatenation of r and s, - * whereas Mbed TLS uses the ASN.1 representation (SEQUENCE of two INTEGERs). - * Add the overhead of ASN.1: up to (1+2) + 2 * (1+2+1) for the - * types, lengths (represented by up to 2 bytes), and potential leading - * zeros of the INTEGERs and the SEQUENCE. */ -#undef MBEDTLS_PK_SIGNATURE_MAX_SIZE -#define MBEDTLS_PK_SIGNATURE_MAX_SIZE ( PSA_VENDOR_ECDSA_SIGNATURE_MAX_SIZE + 11 ) -#endif -#endif /* defined(MBEDTLS_USE_PSA_CRYPTO) */ - -/** - * \brief Types for interfacing with the debug module - */ -typedef enum -{ - MBEDTLS_PK_DEBUG_NONE = 0, - MBEDTLS_PK_DEBUG_MPI, - MBEDTLS_PK_DEBUG_ECP, -} mbedtls_pk_debug_type; - -/** - * \brief Item to send to the debug module - */ -typedef struct mbedtls_pk_debug_item -{ - mbedtls_pk_debug_type MBEDTLS_PRIVATE(type); - const char *MBEDTLS_PRIVATE(name); - void *MBEDTLS_PRIVATE(value); -} mbedtls_pk_debug_item; - -/** Maximum number of item send for debugging, plus 1 */ -#define MBEDTLS_PK_DEBUG_MAX_ITEMS 3 - -/** - * \brief Public key information and operations - */ -typedef struct mbedtls_pk_info_t mbedtls_pk_info_t; - -/** - * \brief Public key container - */ -typedef struct mbedtls_pk_context -{ - const mbedtls_pk_info_t * MBEDTLS_PRIVATE(pk_info); /**< Public key information */ - void * MBEDTLS_PRIVATE(pk_ctx); /**< Underlying public key context */ -} mbedtls_pk_context; - -#if defined(MBEDTLS_ECDSA_C) && defined(MBEDTLS_ECP_RESTARTABLE) -/** - * \brief Context for resuming operations - */ -typedef struct -{ - const mbedtls_pk_info_t * MBEDTLS_PRIVATE(pk_info); /**< Public key information */ - void * MBEDTLS_PRIVATE(rs_ctx); /**< Underlying restart context */ -} mbedtls_pk_restart_ctx; -#else /* MBEDTLS_ECDSA_C && MBEDTLS_ECP_RESTARTABLE */ -/* Now we can declare functions that take a pointer to that */ -typedef void mbedtls_pk_restart_ctx; -#endif /* MBEDTLS_ECDSA_C && MBEDTLS_ECP_RESTARTABLE */ - -#if defined(MBEDTLS_RSA_C) -/** - * Quick access to an RSA context inside a PK context. - * - * \warning You must make sure the PK context actually holds an RSA context - * before using this function! - */ -static inline mbedtls_rsa_context *mbedtls_pk_rsa( const mbedtls_pk_context pk ) -{ - return( (mbedtls_rsa_context *) (pk).MBEDTLS_PRIVATE(pk_ctx) ); -} -#endif /* MBEDTLS_RSA_C */ - -#if defined(MBEDTLS_ECP_C) -/** - * Quick access to an EC context inside a PK context. - * - * \warning You must make sure the PK context actually holds an EC context - * before using this function! - */ -static inline mbedtls_ecp_keypair *mbedtls_pk_ec( const mbedtls_pk_context pk ) -{ - return( (mbedtls_ecp_keypair *) (pk).MBEDTLS_PRIVATE(pk_ctx) ); -} -#endif /* MBEDTLS_ECP_C */ - -#if defined(MBEDTLS_PK_RSA_ALT_SUPPORT) -/** - * \brief Types for RSA-alt abstraction - */ -typedef int (*mbedtls_pk_rsa_alt_decrypt_func)( void *ctx, size_t *olen, - const unsigned char *input, unsigned char *output, - size_t output_max_len ); -typedef int (*mbedtls_pk_rsa_alt_sign_func)( void *ctx, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng, - mbedtls_md_type_t md_alg, unsigned int hashlen, - const unsigned char *hash, unsigned char *sig ); -typedef size_t (*mbedtls_pk_rsa_alt_key_len_func)( void *ctx ); -#endif /* MBEDTLS_PK_RSA_ALT_SUPPORT */ - -/** - * \brief Return information associated with the given PK type - * - * \param pk_type PK type to search for. - * - * \return The PK info associated with the type or NULL if not found. - */ -const mbedtls_pk_info_t *mbedtls_pk_info_from_type( mbedtls_pk_type_t pk_type ); - -/** - * \brief Initialize a #mbedtls_pk_context (as NONE). - * - * \param ctx The context to initialize. - * This must not be \c NULL. - */ -void mbedtls_pk_init( mbedtls_pk_context *ctx ); - -/** - * \brief Free the components of a #mbedtls_pk_context. - * - * \param ctx The context to clear. It must have been initialized. - * If this is \c NULL, this function does nothing. - * - * \note For contexts that have been set up with - * mbedtls_pk_setup_opaque(), this does not free the underlying - * PSA key and you still need to call psa_destroy_key() - * independently if you want to destroy that key. - */ -void mbedtls_pk_free( mbedtls_pk_context *ctx ); - -#if defined(MBEDTLS_ECDSA_C) && defined(MBEDTLS_ECP_RESTARTABLE) -/** - * \brief Initialize a restart context - * - * \param ctx The context to initialize. - * This must not be \c NULL. - */ -void mbedtls_pk_restart_init( mbedtls_pk_restart_ctx *ctx ); - -/** - * \brief Free the components of a restart context - * - * \param ctx The context to clear. It must have been initialized. - * If this is \c NULL, this function does nothing. - */ -void mbedtls_pk_restart_free( mbedtls_pk_restart_ctx *ctx ); -#endif /* MBEDTLS_ECDSA_C && MBEDTLS_ECP_RESTARTABLE */ - -/** - * \brief Initialize a PK context with the information given - * and allocates the type-specific PK subcontext. - * - * \param ctx Context to initialize. It must not have been set - * up yet (type #MBEDTLS_PK_NONE). - * \param info Information to use - * - * \return 0 on success, - * MBEDTLS_ERR_PK_BAD_INPUT_DATA on invalid input, - * MBEDTLS_ERR_PK_ALLOC_FAILED on allocation failure. - * - * \note For contexts holding an RSA-alt key, use - * \c mbedtls_pk_setup_rsa_alt() instead. - */ -int mbedtls_pk_setup( mbedtls_pk_context *ctx, const mbedtls_pk_info_t *info ); - -#if defined(MBEDTLS_USE_PSA_CRYPTO) -/** - * \brief Initialize a PK context to wrap a PSA key. - * - * \note This function replaces mbedtls_pk_setup() for contexts - * that wrap a (possibly opaque) PSA key instead of - * storing and manipulating the key material directly. - * - * \param ctx The context to initialize. It must be empty (type NONE). - * \param key The PSA key to wrap, which must hold an ECC key pair - * (see notes below). - * - * \note The wrapped key must remain valid as long as the - * wrapping PK context is in use, that is at least between - * the point this function is called and the point - * mbedtls_pk_free() is called on this context. The wrapped - * key might then be independently used or destroyed. - * - * \note This function is currently only available for ECC key - * pairs (that is, ECC keys containing private key material). - * Support for other key types may be added later. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_PK_BAD_INPUT_DATA on invalid input - * (context already used, invalid key identifier). - * \return #MBEDTLS_ERR_PK_FEATURE_UNAVAILABLE if the key is not an - * ECC key pair. - * \return #MBEDTLS_ERR_PK_ALLOC_FAILED on allocation failure. - */ -int mbedtls_pk_setup_opaque( mbedtls_pk_context *ctx, - const psa_key_id_t key ); -#endif /* MBEDTLS_USE_PSA_CRYPTO */ - -#if defined(MBEDTLS_PK_RSA_ALT_SUPPORT) -/** - * \brief Initialize an RSA-alt context - * - * \param ctx Context to initialize. It must not have been set - * up yet (type #MBEDTLS_PK_NONE). - * \param key RSA key pointer - * \param decrypt_func Decryption function - * \param sign_func Signing function - * \param key_len_func Function returning key length in bytes - * - * \return 0 on success, or MBEDTLS_ERR_PK_BAD_INPUT_DATA if the - * context wasn't already initialized as RSA_ALT. - * - * \note This function replaces \c mbedtls_pk_setup() for RSA-alt. - */ -int mbedtls_pk_setup_rsa_alt( mbedtls_pk_context *ctx, void * key, - mbedtls_pk_rsa_alt_decrypt_func decrypt_func, - mbedtls_pk_rsa_alt_sign_func sign_func, - mbedtls_pk_rsa_alt_key_len_func key_len_func ); -#endif /* MBEDTLS_PK_RSA_ALT_SUPPORT */ - -/** - * \brief Get the size in bits of the underlying key - * - * \param ctx The context to query. It must have been initialized. - * - * \return Key size in bits, or 0 on error - */ -size_t mbedtls_pk_get_bitlen( const mbedtls_pk_context *ctx ); - -/** - * \brief Get the length in bytes of the underlying key - * - * \param ctx The context to query. It must have been initialized. - * - * \return Key length in bytes, or 0 on error - */ -static inline size_t mbedtls_pk_get_len( const mbedtls_pk_context *ctx ) -{ - return( ( mbedtls_pk_get_bitlen( ctx ) + 7 ) / 8 ); -} - -/** - * \brief Tell if a context can do the operation given by type - * - * \param ctx The context to query. It must have been initialized. - * \param type The desired type. - * - * \return 1 if the context can do operations on the given type. - * \return 0 if the context cannot do the operations on the given - * type. This is always the case for a context that has - * been initialized but not set up, or that has been - * cleared with mbedtls_pk_free(). - */ -int mbedtls_pk_can_do( const mbedtls_pk_context *ctx, mbedtls_pk_type_t type ); - -/** - * \brief Verify signature (including padding if relevant). - * - * \param ctx The PK context to use. It must have been set up. - * \param md_alg Hash algorithm used. - * This can be #MBEDTLS_MD_NONE if the signature algorithm - * does not rely on a hash algorithm (non-deterministic - * ECDSA, RSA PKCS#1 v1.5). - * For PKCS#1 v1.5, if \p md_alg is #MBEDTLS_MD_NONE, then - * \p hash is the DigestInfo structure used by RFC 8017 - * §9.2 steps 3–6. If \p md_alg is a valid hash - * algorithm then \p hash is the digest itself, and this - * function calculates the DigestInfo encoding internally. - * \param hash Hash of the message to sign - * \param hash_len Hash length - * \param sig Signature to verify - * \param sig_len Signature length - * - * \return 0 on success (signature is valid), - * #MBEDTLS_ERR_PK_SIG_LEN_MISMATCH if there is a valid - * signature in sig but its length is less than \p siglen, - * or a specific error code. - * - * \note For RSA keys, the default padding type is PKCS#1 v1.5. - * Use \c mbedtls_pk_verify_ext( MBEDTLS_PK_RSASSA_PSS, ... ) - * to verify RSASSA_PSS signatures. - */ -int mbedtls_pk_verify( mbedtls_pk_context *ctx, mbedtls_md_type_t md_alg, - const unsigned char *hash, size_t hash_len, - const unsigned char *sig, size_t sig_len ); - -/** - * \brief Restartable version of \c mbedtls_pk_verify() - * - * \note Performs the same job as \c mbedtls_pk_verify(), but can - * return early and restart according to the limit set with - * \c mbedtls_ecp_set_max_ops() to reduce blocking for ECC - * operations. For RSA, same as \c mbedtls_pk_verify(). - * - * \param ctx The PK context to use. It must have been set up. - * \param md_alg Hash algorithm used (see notes) - * \param hash Hash of the message to sign - * \param hash_len Hash length or 0 (see notes) - * \param sig Signature to verify - * \param sig_len Signature length - * \param rs_ctx Restart context (NULL to disable restart) - * - * \return See \c mbedtls_pk_verify(), or - * \return #MBEDTLS_ERR_ECP_IN_PROGRESS if maximum number of - * operations was reached: see \c mbedtls_ecp_set_max_ops(). - */ -int mbedtls_pk_verify_restartable( mbedtls_pk_context *ctx, - mbedtls_md_type_t md_alg, - const unsigned char *hash, size_t hash_len, - const unsigned char *sig, size_t sig_len, - mbedtls_pk_restart_ctx *rs_ctx ); - -/** - * \brief Verify signature, with options. - * (Includes verification of the padding depending on type.) - * - * \param type Signature type (inc. possible padding type) to verify - * \param options Pointer to type-specific options, or NULL - * \param ctx The PK context to use. It must have been set up. - * \param md_alg Hash algorithm used (see notes) - * \param hash Hash of the message to sign - * \param hash_len Hash length or 0 (see notes) - * \param sig Signature to verify - * \param sig_len Signature length - * - * \return 0 on success (signature is valid), - * #MBEDTLS_ERR_PK_TYPE_MISMATCH if the PK context can't be - * used for this type of signatures, - * #MBEDTLS_ERR_PK_SIG_LEN_MISMATCH if there is a valid - * signature in sig but its length is less than \p siglen, - * or a specific error code. - * - * \note If hash_len is 0, then the length associated with md_alg - * is used instead, or an error returned if it is invalid. - * - * \note md_alg may be MBEDTLS_MD_NONE, only if hash_len != 0 - * - * \note If type is MBEDTLS_PK_RSASSA_PSS, then options must point - * to a mbedtls_pk_rsassa_pss_options structure, - * otherwise it must be NULL. - */ -int mbedtls_pk_verify_ext( mbedtls_pk_type_t type, const void *options, - mbedtls_pk_context *ctx, mbedtls_md_type_t md_alg, - const unsigned char *hash, size_t hash_len, - const unsigned char *sig, size_t sig_len ); - -/** - * \brief Make signature, including padding if relevant. - * - * \param ctx The PK context to use. It must have been set up - * with a private key. - * \param md_alg Hash algorithm used (see notes) - * \param hash Hash of the message to sign - * \param hash_len Hash length - * \param sig Place to write the signature. - * It must have enough room for the signature. - * #MBEDTLS_PK_SIGNATURE_MAX_SIZE is always enough. - * You may use a smaller buffer if it is large enough - * given the key type. - * \param sig_size The size of the \p sig buffer in bytes. - * \param sig_len On successful return, - * the number of bytes written to \p sig. - * \param f_rng RNG function, must not be \c NULL. - * \param p_rng RNG parameter - * - * \return 0 on success, or a specific error code. - * - * \note For RSA keys, the default padding type is PKCS#1 v1.5. - * There is no interface in the PK module to make RSASSA-PSS - * signatures yet. - * - * \note For RSA, md_alg may be MBEDTLS_MD_NONE if hash_len != 0. - * For ECDSA, md_alg may never be MBEDTLS_MD_NONE. - */ -int mbedtls_pk_sign( mbedtls_pk_context *ctx, mbedtls_md_type_t md_alg, - const unsigned char *hash, size_t hash_len, - unsigned char *sig, size_t sig_size, size_t *sig_len, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng ); - -/** - * \brief Restartable version of \c mbedtls_pk_sign() - * - * \note Performs the same job as \c mbedtls_pk_sign(), but can - * return early and restart according to the limit set with - * \c mbedtls_ecp_set_max_ops() to reduce blocking for ECC - * operations. For RSA, same as \c mbedtls_pk_sign(). - * - * \param ctx The PK context to use. It must have been set up - * with a private key. - * \param md_alg Hash algorithm used (see notes for mbedtls_pk_sign()) - * \param hash Hash of the message to sign - * \param hash_len Hash length - * \param sig Place to write the signature. - * It must have enough room for the signature. - * #MBEDTLS_PK_SIGNATURE_MAX_SIZE is always enough. - * You may use a smaller buffer if it is large enough - * given the key type. - * \param sig_size The size of the \p sig buffer in bytes. - * \param sig_len On successful return, - * the number of bytes written to \p sig. - * \param f_rng RNG function, must not be \c NULL. - * \param p_rng RNG parameter - * \param rs_ctx Restart context (NULL to disable restart) - * - * \return See \c mbedtls_pk_sign(). - * \return #MBEDTLS_ERR_ECP_IN_PROGRESS if maximum number of - * operations was reached: see \c mbedtls_ecp_set_max_ops(). - */ -int mbedtls_pk_sign_restartable( mbedtls_pk_context *ctx, - mbedtls_md_type_t md_alg, - const unsigned char *hash, size_t hash_len, - unsigned char *sig, size_t sig_size, size_t *sig_len, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng, - mbedtls_pk_restart_ctx *rs_ctx ); - -/** - * \brief Decrypt message (including padding if relevant). - * - * \param ctx The PK context to use. It must have been set up - * with a private key. - * \param input Input to decrypt - * \param ilen Input size - * \param output Decrypted output - * \param olen Decrypted message length - * \param osize Size of the output buffer - * \param f_rng RNG function, must not be \c NULL. - * \param p_rng RNG parameter - * - * \note For RSA keys, the default padding type is PKCS#1 v1.5. - * - * \return 0 on success, or a specific error code. - */ -int mbedtls_pk_decrypt( mbedtls_pk_context *ctx, - const unsigned char *input, size_t ilen, - unsigned char *output, size_t *olen, size_t osize, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng ); - -/** - * \brief Encrypt message (including padding if relevant). - * - * \param ctx The PK context to use. It must have been set up. - * \param input Message to encrypt - * \param ilen Message size - * \param output Encrypted output - * \param olen Encrypted output length - * \param osize Size of the output buffer - * \param f_rng RNG function, must not be \c NULL. - * \param p_rng RNG parameter - * - * \note \p f_rng is used for padding generation. - * - * \note For RSA keys, the default padding type is PKCS#1 v1.5. - * - * \return 0 on success, or a specific error code. - */ -int mbedtls_pk_encrypt( mbedtls_pk_context *ctx, - const unsigned char *input, size_t ilen, - unsigned char *output, size_t *olen, size_t osize, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng ); - -/** - * \brief Check if a public-private pair of keys matches. - * - * \param pub Context holding a public key. - * \param prv Context holding a private (and public) key. - * \param f_rng RNG function, must not be \c NULL. - * \param p_rng RNG parameter - * - * \return \c 0 on success (keys were checked and match each other). - * \return #MBEDTLS_ERR_PK_FEATURE_UNAVAILABLE if the keys could not - * be checked - in that case they may or may not match. - * \return #MBEDTLS_ERR_PK_BAD_INPUT_DATA if a context is invalid. - * \return Another non-zero value if the keys do not match. - */ -int mbedtls_pk_check_pair( const mbedtls_pk_context *pub, - const mbedtls_pk_context *prv, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief Export debug information - * - * \param ctx The PK context to use. It must have been initialized. - * \param items Place to write debug items - * - * \return 0 on success or MBEDTLS_ERR_PK_BAD_INPUT_DATA - */ -int mbedtls_pk_debug( const mbedtls_pk_context *ctx, mbedtls_pk_debug_item *items ); - -/** - * \brief Access the type name - * - * \param ctx The PK context to use. It must have been initialized. - * - * \return Type name on success, or "invalid PK" - */ -const char * mbedtls_pk_get_name( const mbedtls_pk_context *ctx ); - -/** - * \brief Get the key type - * - * \param ctx The PK context to use. It must have been initialized. - * - * \return Type on success. - * \return #MBEDTLS_PK_NONE for a context that has not been set up. - */ -mbedtls_pk_type_t mbedtls_pk_get_type( const mbedtls_pk_context *ctx ); - -#if defined(MBEDTLS_PK_PARSE_C) -/** \ingroup pk_module */ -/** - * \brief Parse a private key in PEM or DER format - * - * \param ctx The PK context to fill. It must have been initialized - * but not set up. - * \param key Input buffer to parse. - * The buffer must contain the input exactly, with no - * extra trailing material. For PEM, the buffer must - * contain a null-terminated string. - * \param keylen Size of \b key in bytes. - * For PEM data, this includes the terminating null byte, - * so \p keylen must be equal to `strlen(key) + 1`. - * \param pwd Optional password for decryption. - * Pass \c NULL if expecting a non-encrypted key. - * Pass a string of \p pwdlen bytes if expecting an encrypted - * key; a non-encrypted key will also be accepted. - * The empty password is not supported. - * \param pwdlen Size of the password in bytes. - * Ignored if \p pwd is \c NULL. - * \param f_rng RNG function, must not be \c NULL. Used for blinding. - * \param p_rng RNG parameter - * - * \note On entry, ctx must be empty, either freshly initialised - * with mbedtls_pk_init() or reset with mbedtls_pk_free(). If you need a - * specific key type, check the result with mbedtls_pk_can_do(). - * - * \note The key is also checked for correctness. - * - * \return 0 if successful, or a specific PK or PEM error code - */ -int mbedtls_pk_parse_key( mbedtls_pk_context *ctx, - const unsigned char *key, size_t keylen, - const unsigned char *pwd, size_t pwdlen, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng ); - -/** \ingroup pk_module */ -/** - * \brief Parse a public key in PEM or DER format - * - * \param ctx The PK context to fill. It must have been initialized - * but not set up. - * \param key Input buffer to parse. - * The buffer must contain the input exactly, with no - * extra trailing material. For PEM, the buffer must - * contain a null-terminated string. - * \param keylen Size of \b key in bytes. - * For PEM data, this includes the terminating null byte, - * so \p keylen must be equal to `strlen(key) + 1`. - * - * \note On entry, ctx must be empty, either freshly initialised - * with mbedtls_pk_init() or reset with mbedtls_pk_free(). If you need a - * specific key type, check the result with mbedtls_pk_can_do(). - * - * \note The key is also checked for correctness. - * - * \return 0 if successful, or a specific PK or PEM error code - */ -int mbedtls_pk_parse_public_key( mbedtls_pk_context *ctx, - const unsigned char *key, size_t keylen ); - -#if defined(MBEDTLS_FS_IO) -/** \ingroup pk_module */ -/** - * \brief Load and parse a private key - * - * \param ctx The PK context to fill. It must have been initialized - * but not set up. - * \param path filename to read the private key from - * \param password Optional password to decrypt the file. - * Pass \c NULL if expecting a non-encrypted key. - * Pass a null-terminated string if expecting an encrypted - * key; a non-encrypted key will also be accepted. - * The empty password is not supported. - * \param f_rng RNG function, must not be \c NULL. Used for blinding. - * \param p_rng RNG parameter - * - * \note On entry, ctx must be empty, either freshly initialised - * with mbedtls_pk_init() or reset with mbedtls_pk_free(). If you need a - * specific key type, check the result with mbedtls_pk_can_do(). - * - * \note The key is also checked for correctness. - * - * \return 0 if successful, or a specific PK or PEM error code - */ -int mbedtls_pk_parse_keyfile( mbedtls_pk_context *ctx, - const char *path, const char *password, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng ); - -/** \ingroup pk_module */ -/** - * \brief Load and parse a public key - * - * \param ctx The PK context to fill. It must have been initialized - * but not set up. - * \param path filename to read the public key from - * - * \note On entry, ctx must be empty, either freshly initialised - * with mbedtls_pk_init() or reset with mbedtls_pk_free(). If - * you need a specific key type, check the result with - * mbedtls_pk_can_do(). - * - * \note The key is also checked for correctness. - * - * \return 0 if successful, or a specific PK or PEM error code - */ -int mbedtls_pk_parse_public_keyfile( mbedtls_pk_context *ctx, const char *path ); -#endif /* MBEDTLS_FS_IO */ -#endif /* MBEDTLS_PK_PARSE_C */ - -#if defined(MBEDTLS_PK_WRITE_C) -/** - * \brief Write a private key to a PKCS#1 or SEC1 DER structure - * Note: data is written at the end of the buffer! Use the - * return value to determine where you should start - * using the buffer - * - * \param ctx PK context which must contain a valid private key. - * \param buf buffer to write to - * \param size size of the buffer - * - * \return length of data written if successful, or a specific - * error code - */ -int mbedtls_pk_write_key_der( const mbedtls_pk_context *ctx, unsigned char *buf, size_t size ); - -/** - * \brief Write a public key to a SubjectPublicKeyInfo DER structure - * Note: data is written at the end of the buffer! Use the - * return value to determine where you should start - * using the buffer - * - * \param ctx PK context which must contain a valid public or private key. - * \param buf buffer to write to - * \param size size of the buffer - * - * \return length of data written if successful, or a specific - * error code - */ -int mbedtls_pk_write_pubkey_der( const mbedtls_pk_context *ctx, unsigned char *buf, size_t size ); - -#if defined(MBEDTLS_PEM_WRITE_C) -/** - * \brief Write a public key to a PEM string - * - * \param ctx PK context which must contain a valid public or private key. - * \param buf Buffer to write to. The output includes a - * terminating null byte. - * \param size Size of the buffer in bytes. - * - * \return 0 if successful, or a specific error code - */ -int mbedtls_pk_write_pubkey_pem( const mbedtls_pk_context *ctx, unsigned char *buf, size_t size ); - -/** - * \brief Write a private key to a PKCS#1 or SEC1 PEM string - * - * \param ctx PK context which must contain a valid private key. - * \param buf Buffer to write to. The output includes a - * terminating null byte. - * \param size Size of the buffer in bytes. - * - * \return 0 if successful, or a specific error code - */ -int mbedtls_pk_write_key_pem( const mbedtls_pk_context *ctx, unsigned char *buf, size_t size ); -#endif /* MBEDTLS_PEM_WRITE_C */ -#endif /* MBEDTLS_PK_WRITE_C */ - -/* - * WARNING: Low-level functions. You probably do not want to use these unless - * you are certain you do ;) - */ - -#if defined(MBEDTLS_PK_PARSE_C) -/** - * \brief Parse a SubjectPublicKeyInfo DER structure - * - * \param p the position in the ASN.1 data - * \param end end of the buffer - * \param pk The PK context to fill. It must have been initialized - * but not set up. - * - * \return 0 if successful, or a specific PK error code - */ -int mbedtls_pk_parse_subpubkey( unsigned char **p, const unsigned char *end, - mbedtls_pk_context *pk ); -#endif /* MBEDTLS_PK_PARSE_C */ - -#if defined(MBEDTLS_PK_WRITE_C) -/** - * \brief Write a subjectPublicKey to ASN.1 data - * Note: function works backwards in data buffer - * - * \param p reference to current position pointer - * \param start start of the buffer (for bounds-checking) - * \param key PK context which must contain a valid public or private key. - * - * \return the length written or a negative error code - */ -int mbedtls_pk_write_pubkey( unsigned char **p, unsigned char *start, - const mbedtls_pk_context *key ); -#endif /* MBEDTLS_PK_WRITE_C */ - -/* - * Internal module functions. You probably do not want to use these unless you - * know you do. - */ -#if defined(MBEDTLS_FS_IO) -int mbedtls_pk_load_file( const char *path, unsigned char **buf, size_t *n ); -#endif - -#if defined(MBEDTLS_USE_PSA_CRYPTO) -/** - * \brief Turn an EC key into an opaque one. - * - * \warning This is a temporary utility function for tests. It might - * change or be removed at any time without notice. - * - * \note Only ECDSA keys are supported so far. Signing with the - * specified hash is the only allowed use of that key. - * - * \param pk Input: the EC key to import to a PSA key. - * Output: a PK context wrapping that PSA key. - * \param key Output: a PSA key identifier. - * It's the caller's responsibility to call - * psa_destroy_key() on that key identifier after calling - * mbedtls_pk_free() on the PK context. - * \param hash_alg The hash algorithm to allow for use with that key. - * - * \return \c 0 if successful. - * \return An Mbed TLS error code otherwise. - */ -int mbedtls_pk_wrap_as_opaque( mbedtls_pk_context *pk, - psa_key_id_t *key, - psa_algorithm_t hash_alg ); -#endif /* MBEDTLS_USE_PSA_CRYPTO */ - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_PK_H */ diff --git a/windows/mbedtls/pkcs12.h b/windows/mbedtls/pkcs12.h deleted file mode 100644 index de139ac6..00000000 --- a/windows/mbedtls/pkcs12.h +++ /dev/null @@ -1,105 +0,0 @@ -/** - * \file pkcs12.h - * - * \brief PKCS#12 Personal Information Exchange Syntax - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_PKCS12_H -#define MBEDTLS_PKCS12_H - -#include "mbedtls/build_info.h" - -#include "mbedtls/md.h" -#include "mbedtls/cipher.h" -#include "mbedtls/asn1.h" - -#include - -#define MBEDTLS_ERR_PKCS12_BAD_INPUT_DATA -0x1F80 /**< Bad input parameters to function. */ -#define MBEDTLS_ERR_PKCS12_FEATURE_UNAVAILABLE -0x1F00 /**< Feature not available, e.g. unsupported encryption scheme. */ -#define MBEDTLS_ERR_PKCS12_PBE_INVALID_FORMAT -0x1E80 /**< PBE ASN.1 data not as expected. */ -#define MBEDTLS_ERR_PKCS12_PASSWORD_MISMATCH -0x1E00 /**< Given private key password does not allow for correct decryption. */ - -#define MBEDTLS_PKCS12_DERIVE_KEY 1 /**< encryption/decryption key */ -#define MBEDTLS_PKCS12_DERIVE_IV 2 /**< initialization vector */ -#define MBEDTLS_PKCS12_DERIVE_MAC_KEY 3 /**< integrity / MAC key */ - -#define MBEDTLS_PKCS12_PBE_DECRYPT 0 -#define MBEDTLS_PKCS12_PBE_ENCRYPT 1 - -#ifdef __cplusplus -extern "C" { -#endif - -#if defined(MBEDTLS_ASN1_PARSE_C) - -/** - * \brief PKCS12 Password Based function (encryption / decryption) - * for cipher-based and mbedtls_md-based PBE's - * - * \param pbe_params an ASN1 buffer containing the pkcs-12PbeParams structure - * \param mode either MBEDTLS_PKCS12_PBE_ENCRYPT or MBEDTLS_PKCS12_PBE_DECRYPT - * \param cipher_type the cipher used - * \param md_type the mbedtls_md used - * \param pwd the password used (may be NULL if no password is used) - * \param pwdlen length of the password (may be 0) - * \param input the input data - * \param len data length - * \param output the output buffer - * - * \return 0 if successful, or a MBEDTLS_ERR_XXX code - */ -int mbedtls_pkcs12_pbe( mbedtls_asn1_buf *pbe_params, int mode, - mbedtls_cipher_type_t cipher_type, mbedtls_md_type_t md_type, - const unsigned char *pwd, size_t pwdlen, - const unsigned char *input, size_t len, - unsigned char *output ); - -#endif /* MBEDTLS_ASN1_PARSE_C */ - -/** - * \brief The PKCS#12 derivation function uses a password and a salt - * to produce pseudo-random bits for a particular "purpose". - * - * Depending on the given id, this function can produce an - * encryption/decryption key, an nitialization vector or an - * integrity key. - * - * \param data buffer to store the derived data in - * \param datalen length to fill - * \param pwd password to use (may be NULL if no password is used) - * \param pwdlen length of the password (may be 0) - * \param salt salt buffer to use - * \param saltlen length of the salt - * \param mbedtls_md mbedtls_md type to use during the derivation - * \param id id that describes the purpose (can be MBEDTLS_PKCS12_DERIVE_KEY, - * MBEDTLS_PKCS12_DERIVE_IV or MBEDTLS_PKCS12_DERIVE_MAC_KEY) - * \param iterations number of iterations - * - * \return 0 if successful, or a MD, BIGNUM type error. - */ -int mbedtls_pkcs12_derivation( unsigned char *data, size_t datalen, - const unsigned char *pwd, size_t pwdlen, - const unsigned char *salt, size_t saltlen, - mbedtls_md_type_t mbedtls_md, int id, int iterations ); - -#ifdef __cplusplus -} -#endif - -#endif /* pkcs12.h */ diff --git a/windows/mbedtls/pkcs5.h b/windows/mbedtls/pkcs5.h deleted file mode 100644 index c339398a..00000000 --- a/windows/mbedtls/pkcs5.h +++ /dev/null @@ -1,103 +0,0 @@ -/** - * \file pkcs5.h - * - * \brief PKCS#5 functions - * - * \author Mathias Olsson - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_PKCS5_H -#define MBEDTLS_PKCS5_H - -#include "mbedtls/build_info.h" - -#include "mbedtls/asn1.h" -#include "mbedtls/md.h" - -#include -#include - -#define MBEDTLS_ERR_PKCS5_BAD_INPUT_DATA -0x2f80 /**< Bad input parameters to function. */ -#define MBEDTLS_ERR_PKCS5_INVALID_FORMAT -0x2f00 /**< Unexpected ASN.1 data. */ -#define MBEDTLS_ERR_PKCS5_FEATURE_UNAVAILABLE -0x2e80 /**< Requested encryption or digest alg not available. */ -#define MBEDTLS_ERR_PKCS5_PASSWORD_MISMATCH -0x2e00 /**< Given private key password does not allow for correct decryption. */ - -#define MBEDTLS_PKCS5_DECRYPT 0 -#define MBEDTLS_PKCS5_ENCRYPT 1 - -#ifdef __cplusplus -extern "C" { -#endif - -#if defined(MBEDTLS_ASN1_PARSE_C) - -/** - * \brief PKCS#5 PBES2 function - * - * \param pbe_params the ASN.1 algorithm parameters - * \param mode either MBEDTLS_PKCS5_DECRYPT or MBEDTLS_PKCS5_ENCRYPT - * \param pwd password to use when generating key - * \param pwdlen length of password - * \param data data to process - * \param datalen length of data - * \param output output buffer - * - * \returns 0 on success, or a MBEDTLS_ERR_XXX code if verification fails. - */ -int mbedtls_pkcs5_pbes2( const mbedtls_asn1_buf *pbe_params, int mode, - const unsigned char *pwd, size_t pwdlen, - const unsigned char *data, size_t datalen, - unsigned char *output ); - -#endif /* MBEDTLS_ASN1_PARSE_C */ - -/** - * \brief PKCS#5 PBKDF2 using HMAC - * - * \param ctx Generic HMAC context - * \param password Password to use when generating key - * \param plen Length of password - * \param salt Salt to use when generating key - * \param slen Length of salt - * \param iteration_count Iteration count - * \param key_length Length of generated key in bytes - * \param output Generated key. Must be at least as big as key_length - * - * \returns 0 on success, or a MBEDTLS_ERR_XXX code if verification fails. - */ -int mbedtls_pkcs5_pbkdf2_hmac( mbedtls_md_context_t *ctx, const unsigned char *password, - size_t plen, const unsigned char *salt, size_t slen, - unsigned int iteration_count, - uint32_t key_length, unsigned char *output ); - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief Checkup routine - * - * \return 0 if successful, or 1 if the test failed - */ -int mbedtls_pkcs5_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* pkcs5.h */ diff --git a/windows/mbedtls/platform.h b/windows/mbedtls/platform.h deleted file mode 100644 index 277a85c7..00000000 --- a/windows/mbedtls/platform.h +++ /dev/null @@ -1,411 +0,0 @@ -/** - * \file platform.h - * - * \brief This file contains the definitions and functions of the - * Mbed TLS platform abstraction layer. - * - * The platform abstraction layer removes the need for the library - * to directly link to standard C library functions or operating - * system services, making the library easier to port and embed. - * Application developers and users of the library can provide their own - * implementations of these functions, or implementations specific to - * their platform, which can be statically linked to the library or - * dynamically configured at runtime. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_PLATFORM_H -#define MBEDTLS_PLATFORM_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#if defined(MBEDTLS_HAVE_TIME) -#include "mbedtls/platform_time.h" -#endif - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \name SECTION: Module settings - * - * The configuration options you can set for this module are in this section. - * Either change them in mbedtls_config.h or define them on the compiler command line. - * \{ - */ - -/* The older Microsoft Windows common runtime provides non-conforming - * implementations of some standard library functions, including snprintf - * and vsnprintf. This affects MSVC and MinGW builds. - */ -#if defined(__MINGW32__) || (defined(_MSC_VER) && _MSC_VER <= 1900) -#define MBEDTLS_PLATFORM_HAS_NON_CONFORMING_SNPRINTF -#define MBEDTLS_PLATFORM_HAS_NON_CONFORMING_VSNPRINTF -#endif - -#if !defined(MBEDTLS_PLATFORM_NO_STD_FUNCTIONS) -#include -#include -#include -#if !defined(MBEDTLS_PLATFORM_STD_SNPRINTF) -#if defined(MBEDTLS_PLATFORM_HAS_NON_CONFORMING_SNPRINTF) -#define MBEDTLS_PLATFORM_STD_SNPRINTF mbedtls_platform_win32_snprintf /**< The default \c snprintf function to use. */ -#else -#define MBEDTLS_PLATFORM_STD_SNPRINTF snprintf /**< The default \c snprintf function to use. */ -#endif -#endif -#if !defined(MBEDTLS_PLATFORM_STD_VSNPRINTF) -#if defined(MBEDTLS_PLATFORM_HAS_NON_CONFORMING_VSNPRINTF) -#define MBEDTLS_PLATFORM_STD_VSNPRINTF mbedtls_platform_win32_vsnprintf /**< The default \c vsnprintf function to use. */ -#else -#define MBEDTLS_PLATFORM_STD_VSNPRINTF vsnprintf /**< The default \c vsnprintf function to use. */ -#endif -#endif -#if !defined(MBEDTLS_PLATFORM_STD_PRINTF) -#define MBEDTLS_PLATFORM_STD_PRINTF printf /**< The default \c printf function to use. */ -#endif -#if !defined(MBEDTLS_PLATFORM_STD_FPRINTF) -#define MBEDTLS_PLATFORM_STD_FPRINTF fprintf /**< The default \c fprintf function to use. */ -#endif -#if !defined(MBEDTLS_PLATFORM_STD_CALLOC) -#define MBEDTLS_PLATFORM_STD_CALLOC calloc /**< The default \c calloc function to use. */ -#endif -#if !defined(MBEDTLS_PLATFORM_STD_FREE) -#define MBEDTLS_PLATFORM_STD_FREE free /**< The default \c free function to use. */ -#endif -#if !defined(MBEDTLS_PLATFORM_STD_EXIT) -#define MBEDTLS_PLATFORM_STD_EXIT exit /**< The default \c exit function to use. */ -#endif -#if !defined(MBEDTLS_PLATFORM_STD_TIME) -#define MBEDTLS_PLATFORM_STD_TIME time /**< The default \c time function to use. */ -#endif -#if !defined(MBEDTLS_PLATFORM_STD_EXIT_SUCCESS) -#define MBEDTLS_PLATFORM_STD_EXIT_SUCCESS EXIT_SUCCESS /**< The default exit value to use. */ -#endif -#if !defined(MBEDTLS_PLATFORM_STD_EXIT_FAILURE) -#define MBEDTLS_PLATFORM_STD_EXIT_FAILURE EXIT_FAILURE /**< The default exit value to use. */ -#endif -#if defined(MBEDTLS_FS_IO) -#if !defined(MBEDTLS_PLATFORM_STD_NV_SEED_READ) -#define MBEDTLS_PLATFORM_STD_NV_SEED_READ mbedtls_platform_std_nv_seed_read -#endif -#if !defined(MBEDTLS_PLATFORM_STD_NV_SEED_WRITE) -#define MBEDTLS_PLATFORM_STD_NV_SEED_WRITE mbedtls_platform_std_nv_seed_write -#endif -#if !defined(MBEDTLS_PLATFORM_STD_NV_SEED_FILE) -#define MBEDTLS_PLATFORM_STD_NV_SEED_FILE "seedfile" -#endif -#endif /* MBEDTLS_FS_IO */ -#else /* MBEDTLS_PLATFORM_NO_STD_FUNCTIONS */ -#if defined(MBEDTLS_PLATFORM_STD_MEM_HDR) -#include MBEDTLS_PLATFORM_STD_MEM_HDR -#endif -#endif /* MBEDTLS_PLATFORM_NO_STD_FUNCTIONS */ - - -/* \} name SECTION: Module settings */ - -/* - * The function pointers for calloc and free. - */ -#if defined(MBEDTLS_PLATFORM_MEMORY) -#if defined(MBEDTLS_PLATFORM_FREE_MACRO) && \ - defined(MBEDTLS_PLATFORM_CALLOC_MACRO) -#define mbedtls_free MBEDTLS_PLATFORM_FREE_MACRO -#define mbedtls_calloc MBEDTLS_PLATFORM_CALLOC_MACRO -#else -/* For size_t */ -#include -extern void *mbedtls_calloc( size_t n, size_t size ); -extern void mbedtls_free( void *ptr ); - -/** - * \brief This function dynamically sets the memory-management - * functions used by the library, during runtime. - * - * \param calloc_func The \c calloc function implementation. - * \param free_func The \c free function implementation. - * - * \return \c 0. - */ -int mbedtls_platform_set_calloc_free( void * (*calloc_func)( size_t, size_t ), - void (*free_func)( void * ) ); -#endif /* MBEDTLS_PLATFORM_FREE_MACRO && MBEDTLS_PLATFORM_CALLOC_MACRO */ -#else /* !MBEDTLS_PLATFORM_MEMORY */ -#define mbedtls_free free -#define mbedtls_calloc calloc -#endif /* MBEDTLS_PLATFORM_MEMORY && !MBEDTLS_PLATFORM_{FREE,CALLOC}_MACRO */ - -/* - * The function pointers for fprintf - */ -#if defined(MBEDTLS_PLATFORM_FPRINTF_ALT) -/* We need FILE * */ -#include -extern int (*mbedtls_fprintf)( FILE *stream, const char *format, ... ); - -/** - * \brief This function dynamically configures the fprintf - * function that is called when the - * mbedtls_fprintf() function is invoked by the library. - * - * \param fprintf_func The \c fprintf function implementation. - * - * \return \c 0. - */ -int mbedtls_platform_set_fprintf( int (*fprintf_func)( FILE *stream, const char *, - ... ) ); -#else -#if defined(MBEDTLS_PLATFORM_FPRINTF_MACRO) -#define mbedtls_fprintf MBEDTLS_PLATFORM_FPRINTF_MACRO -#else -#define mbedtls_fprintf fprintf -#endif /* MBEDTLS_PLATFORM_FPRINTF_MACRO */ -#endif /* MBEDTLS_PLATFORM_FPRINTF_ALT */ - -/* - * The function pointers for printf - */ -#if defined(MBEDTLS_PLATFORM_PRINTF_ALT) -extern int (*mbedtls_printf)( const char *format, ... ); - -/** - * \brief This function dynamically configures the snprintf - * function that is called when the mbedtls_snprintf() - * function is invoked by the library. - * - * \param printf_func The \c printf function implementation. - * - * \return \c 0 on success. - */ -int mbedtls_platform_set_printf( int (*printf_func)( const char *, ... ) ); -#else /* !MBEDTLS_PLATFORM_PRINTF_ALT */ -#if defined(MBEDTLS_PLATFORM_PRINTF_MACRO) -#define mbedtls_printf MBEDTLS_PLATFORM_PRINTF_MACRO -#else -#define mbedtls_printf printf -#endif /* MBEDTLS_PLATFORM_PRINTF_MACRO */ -#endif /* MBEDTLS_PLATFORM_PRINTF_ALT */ - -/* - * The function pointers for snprintf - * - * The snprintf implementation should conform to C99: - * - it *must* always correctly zero-terminate the buffer - * (except when n == 0, then it must leave the buffer untouched) - * - however it is acceptable to return -1 instead of the required length when - * the destination buffer is too short. - */ -#if defined(MBEDTLS_PLATFORM_HAS_NON_CONFORMING_SNPRINTF) -/* For Windows (inc. MSYS2), we provide our own fixed implementation */ -int mbedtls_platform_win32_snprintf( char *s, size_t n, const char *fmt, ... ); -#endif - -#if defined(MBEDTLS_PLATFORM_SNPRINTF_ALT) -extern int (*mbedtls_snprintf)( char * s, size_t n, const char * format, ... ); - -/** - * \brief This function allows configuring a custom - * \c snprintf function pointer. - * - * \param snprintf_func The \c snprintf function implementation. - * - * \return \c 0 on success. - */ -int mbedtls_platform_set_snprintf( int (*snprintf_func)( char * s, size_t n, - const char * format, ... ) ); -#else /* MBEDTLS_PLATFORM_SNPRINTF_ALT */ -#if defined(MBEDTLS_PLATFORM_SNPRINTF_MACRO) -#define mbedtls_snprintf MBEDTLS_PLATFORM_SNPRINTF_MACRO -#else -#define mbedtls_snprintf MBEDTLS_PLATFORM_STD_SNPRINTF -#endif /* MBEDTLS_PLATFORM_SNPRINTF_MACRO */ -#endif /* MBEDTLS_PLATFORM_SNPRINTF_ALT */ - -/* - * The function pointers for vsnprintf - * - * The vsnprintf implementation should conform to C99: - * - it *must* always correctly zero-terminate the buffer - * (except when n == 0, then it must leave the buffer untouched) - * - however it is acceptable to return -1 instead of the required length when - * the destination buffer is too short. - */ -#if defined(MBEDTLS_PLATFORM_HAS_NON_CONFORMING_VSNPRINTF) -#include -/* For Older Windows (inc. MSYS2), we provide our own fixed implementation */ -int mbedtls_platform_win32_vsnprintf( char *s, size_t n, const char *fmt, va_list arg ); -#endif - -#if defined(MBEDTLS_PLATFORM_VSNPRINTF_ALT) -#include -extern int (*mbedtls_vsnprintf)( char * s, size_t n, const char * format, va_list arg ); - -/** - * \brief Set your own snprintf function pointer - * - * \param vsnprintf_func The \c vsnprintf function implementation - * - * \return \c 0 - */ -int mbedtls_platform_set_vsnprintf( int (*vsnprintf_func)( char * s, size_t n, - const char * format, va_list arg ) ); -#else /* MBEDTLS_PLATFORM_VSNPRINTF_ALT */ -#if defined(MBEDTLS_PLATFORM_VSNPRINTF_MACRO) -#define mbedtls_vsnprintf MBEDTLS_PLATFORM_VSNPRINTF_MACRO -#else -#define mbedtls_vsnprintf vsnprintf -#endif /* MBEDTLS_PLATFORM_VSNPRINTF_MACRO */ -#endif /* MBEDTLS_PLATFORM_VSNPRINTF_ALT */ - -/* - * The function pointers for exit - */ -#if defined(MBEDTLS_PLATFORM_EXIT_ALT) -extern void (*mbedtls_exit)( int status ); - -/** - * \brief This function dynamically configures the exit - * function that is called when the mbedtls_exit() - * function is invoked by the library. - * - * \param exit_func The \c exit function implementation. - * - * \return \c 0 on success. - */ -int mbedtls_platform_set_exit( void (*exit_func)( int status ) ); -#else -#if defined(MBEDTLS_PLATFORM_EXIT_MACRO) -#define mbedtls_exit MBEDTLS_PLATFORM_EXIT_MACRO -#else -#define mbedtls_exit exit -#endif /* MBEDTLS_PLATFORM_EXIT_MACRO */ -#endif /* MBEDTLS_PLATFORM_EXIT_ALT */ - -/* - * The default exit values - */ -#if defined(MBEDTLS_PLATFORM_STD_EXIT_SUCCESS) -#define MBEDTLS_EXIT_SUCCESS MBEDTLS_PLATFORM_STD_EXIT_SUCCESS -#else -#define MBEDTLS_EXIT_SUCCESS 0 -#endif -#if defined(MBEDTLS_PLATFORM_STD_EXIT_FAILURE) -#define MBEDTLS_EXIT_FAILURE MBEDTLS_PLATFORM_STD_EXIT_FAILURE -#else -#define MBEDTLS_EXIT_FAILURE 1 -#endif - -/* - * The function pointers for reading from and writing a seed file to - * Non-Volatile storage (NV) in a platform-independent way - * - * Only enabled when the NV seed entropy source is enabled - */ -#if defined(MBEDTLS_ENTROPY_NV_SEED) -#if !defined(MBEDTLS_PLATFORM_NO_STD_FUNCTIONS) && defined(MBEDTLS_FS_IO) -/* Internal standard platform definitions */ -int mbedtls_platform_std_nv_seed_read( unsigned char *buf, size_t buf_len ); -int mbedtls_platform_std_nv_seed_write( unsigned char *buf, size_t buf_len ); -#endif - -#if defined(MBEDTLS_PLATFORM_NV_SEED_ALT) -extern int (*mbedtls_nv_seed_read)( unsigned char *buf, size_t buf_len ); -extern int (*mbedtls_nv_seed_write)( unsigned char *buf, size_t buf_len ); - -/** - * \brief This function allows configuring custom seed file writing and - * reading functions. - * - * \param nv_seed_read_func The seed reading function implementation. - * \param nv_seed_write_func The seed writing function implementation. - * - * \return \c 0 on success. - */ -int mbedtls_platform_set_nv_seed( - int (*nv_seed_read_func)( unsigned char *buf, size_t buf_len ), - int (*nv_seed_write_func)( unsigned char *buf, size_t buf_len ) - ); -#else -#if defined(MBEDTLS_PLATFORM_NV_SEED_READ_MACRO) && \ - defined(MBEDTLS_PLATFORM_NV_SEED_WRITE_MACRO) -#define mbedtls_nv_seed_read MBEDTLS_PLATFORM_NV_SEED_READ_MACRO -#define mbedtls_nv_seed_write MBEDTLS_PLATFORM_NV_SEED_WRITE_MACRO -#else -#define mbedtls_nv_seed_read mbedtls_platform_std_nv_seed_read -#define mbedtls_nv_seed_write mbedtls_platform_std_nv_seed_write -#endif -#endif /* MBEDTLS_PLATFORM_NV_SEED_ALT */ -#endif /* MBEDTLS_ENTROPY_NV_SEED */ - -#if !defined(MBEDTLS_PLATFORM_SETUP_TEARDOWN_ALT) - -/** - * \brief The platform context structure. - * - * \note This structure may be used to assist platform-specific - * setup or teardown operations. - */ -typedef struct mbedtls_platform_context -{ - char MBEDTLS_PRIVATE(dummy); /**< A placeholder member, as empty structs are not portable. */ -} -mbedtls_platform_context; - -#else -#include "platform_alt.h" -#endif /* !MBEDTLS_PLATFORM_SETUP_TEARDOWN_ALT */ - -/** - * \brief This function performs any platform-specific initialization - * operations. - * - * \note This function should be called before any other library functions. - * - * Its implementation is platform-specific, and unless - * platform-specific code is provided, it does nothing. - * - * \note The usage and necessity of this function is dependent on the platform. - * - * \param ctx The platform context. - * - * \return \c 0 on success. - */ -int mbedtls_platform_setup( mbedtls_platform_context *ctx ); -/** - * \brief This function performs any platform teardown operations. - * - * \note This function should be called after every other Mbed TLS module - * has been correctly freed using the appropriate free function. - * - * Its implementation is platform-specific, and unless - * platform-specific code is provided, it does nothing. - * - * \note The usage and necessity of this function is dependent on the platform. - * - * \param ctx The platform context. - * - */ -void mbedtls_platform_teardown( mbedtls_platform_context *ctx ); - -#ifdef __cplusplus -} -#endif - -#endif /* platform.h */ diff --git a/windows/mbedtls/platform_time.h b/windows/mbedtls/platform_time.h deleted file mode 100644 index 8d4b95d1..00000000 --- a/windows/mbedtls/platform_time.h +++ /dev/null @@ -1,76 +0,0 @@ -/** - * \file platform_time.h - * - * \brief mbed TLS Platform time abstraction - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_PLATFORM_TIME_H -#define MBEDTLS_PLATFORM_TIME_H - -#include "mbedtls/build_info.h" - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \name SECTION: Module settings - * - * The configuration options you can set for this module are in this section. - * Either change them in mbedtls_config.h or define them on the compiler command line. - * \{ - */ - -/* - * The time_t datatype - */ -#if defined(MBEDTLS_PLATFORM_TIME_TYPE_MACRO) -typedef MBEDTLS_PLATFORM_TIME_TYPE_MACRO mbedtls_time_t; -#else -/* For time_t */ -#include -typedef time_t mbedtls_time_t; -#endif /* MBEDTLS_PLATFORM_TIME_TYPE_MACRO */ - -/* - * The function pointers for time - */ -#if defined(MBEDTLS_PLATFORM_TIME_ALT) -extern mbedtls_time_t (*mbedtls_time)( mbedtls_time_t* time ); - -/** - * \brief Set your own time function pointer - * - * \param time_func the time function implementation - * - * \return 0 - */ -int mbedtls_platform_set_time( mbedtls_time_t (*time_func)( mbedtls_time_t* time ) ); -#else -#if defined(MBEDTLS_PLATFORM_TIME_MACRO) -#define mbedtls_time MBEDTLS_PLATFORM_TIME_MACRO -#else -#define mbedtls_time time -#endif /* MBEDTLS_PLATFORM_TIME_MACRO */ -#endif /* MBEDTLS_PLATFORM_TIME_ALT */ - -#ifdef __cplusplus -} -#endif - -#endif /* platform_time.h */ diff --git a/windows/mbedtls/platform_util.h b/windows/mbedtls/platform_util.h deleted file mode 100644 index 1a0a1351..00000000 --- a/windows/mbedtls/platform_util.h +++ /dev/null @@ -1,122 +0,0 @@ -/** - * \file platform_util.h - * - * \brief Common and shared functions used by multiple modules in the Mbed TLS - * library. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_PLATFORM_UTIL_H -#define MBEDTLS_PLATFORM_UTIL_H - -#include "mbedtls/build_info.h" - -#include -#if defined(MBEDTLS_HAVE_TIME_DATE) -#include "mbedtls/platform_time.h" -#include -#endif /* MBEDTLS_HAVE_TIME_DATE */ - -#ifdef __cplusplus -extern "C" { -#endif - -/* Internal macros meant to be called only from within the library. */ -#define MBEDTLS_INTERNAL_VALIDATE_RET( cond, ret ) do { } while( 0 ) -#define MBEDTLS_INTERNAL_VALIDATE( cond ) do { } while( 0 ) - -/* Internal helper macros for deprecating API constants. */ -#if !defined(MBEDTLS_DEPRECATED_REMOVED) -#if defined(MBEDTLS_DEPRECATED_WARNING) -/* Deliberately don't (yet) export MBEDTLS_DEPRECATED here - * to avoid conflict with other headers which define and use - * it, too. We might want to move all these definitions here at - * some point for uniformity. */ -#define MBEDTLS_DEPRECATED __attribute__((deprecated)) -MBEDTLS_DEPRECATED typedef char const * mbedtls_deprecated_string_constant_t; -#define MBEDTLS_DEPRECATED_STRING_CONSTANT( VAL ) \ - ( (mbedtls_deprecated_string_constant_t) ( VAL ) ) -MBEDTLS_DEPRECATED typedef int mbedtls_deprecated_numeric_constant_t; -#define MBEDTLS_DEPRECATED_NUMERIC_CONSTANT( VAL ) \ - ( (mbedtls_deprecated_numeric_constant_t) ( VAL ) ) -#undef MBEDTLS_DEPRECATED -#else /* MBEDTLS_DEPRECATED_WARNING */ -#define MBEDTLS_DEPRECATED_STRING_CONSTANT( VAL ) VAL -#define MBEDTLS_DEPRECATED_NUMERIC_CONSTANT( VAL ) VAL -#endif /* MBEDTLS_DEPRECATED_WARNING */ -#endif /* MBEDTLS_DEPRECATED_REMOVED */ - -/** - * \brief Securely zeroize a buffer - * - * The function is meant to wipe the data contained in a buffer so - * that it can no longer be recovered even if the program memory - * is later compromised. Call this function on sensitive data - * stored on the stack before returning from a function, and on - * sensitive data stored on the heap before freeing the heap - * object. - * - * It is extremely difficult to guarantee that calls to - * mbedtls_platform_zeroize() are not removed by aggressive - * compiler optimizations in a portable way. For this reason, Mbed - * TLS provides the configuration option - * MBEDTLS_PLATFORM_ZEROIZE_ALT, which allows users to configure - * mbedtls_platform_zeroize() to use a suitable implementation for - * their platform and needs - * - * \param buf Buffer to be zeroized - * \param len Length of the buffer in bytes - * - */ -void mbedtls_platform_zeroize( void *buf, size_t len ); - -#if defined(MBEDTLS_HAVE_TIME_DATE) -/** - * \brief Platform-specific implementation of gmtime_r() - * - * The function is a thread-safe abstraction that behaves - * similarly to the gmtime_r() function from Unix/POSIX. - * - * Mbed TLS will try to identify the underlying platform and - * make use of an appropriate underlying implementation (e.g. - * gmtime_r() for POSIX and gmtime_s() for Windows). If this is - * not possible, then gmtime() will be used. In this case, calls - * from the library to gmtime() will be guarded by the mutex - * mbedtls_threading_gmtime_mutex if MBEDTLS_THREADING_C is - * enabled. It is recommended that calls from outside the library - * are also guarded by this mutex. - * - * If MBEDTLS_PLATFORM_GMTIME_R_ALT is defined, then Mbed TLS will - * unconditionally use the alternative implementation for - * mbedtls_platform_gmtime_r() supplied by the user at compile time. - * - * \param tt Pointer to an object containing time (in seconds) since the - * epoch to be converted - * \param tm_buf Pointer to an object where the results will be stored - * - * \return Pointer to an object of type struct tm on success, otherwise - * NULL - */ -struct tm *mbedtls_platform_gmtime_r( const mbedtls_time_t *tt, - struct tm *tm_buf ); -#endif /* MBEDTLS_HAVE_TIME_DATE */ - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_PLATFORM_UTIL_H */ diff --git a/windows/mbedtls/poly1305.h b/windows/mbedtls/poly1305.h deleted file mode 100644 index 7984ca4b..00000000 --- a/windows/mbedtls/poly1305.h +++ /dev/null @@ -1,180 +0,0 @@ -/** - * \file poly1305.h - * - * \brief This file contains Poly1305 definitions and functions. - * - * Poly1305 is a one-time message authenticator that can be used to - * authenticate messages. Poly1305-AES was created by Daniel - * Bernstein https://cr.yp.to/mac/poly1305-20050329.pdf The generic - * Poly1305 algorithm (not tied to AES) was also standardized in RFC - * 7539. - * - * \author Daniel King - */ - -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_POLY1305_H -#define MBEDTLS_POLY1305_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -#define MBEDTLS_ERR_POLY1305_BAD_INPUT_DATA -0x0057 /**< Invalid input parameter(s). */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_POLY1305_ALT) - -typedef struct mbedtls_poly1305_context -{ - uint32_t MBEDTLS_PRIVATE(r)[4]; /** The value for 'r' (low 128 bits of the key). */ - uint32_t MBEDTLS_PRIVATE(s)[4]; /** The value for 's' (high 128 bits of the key). */ - uint32_t MBEDTLS_PRIVATE(acc)[5]; /** The accumulator number. */ - uint8_t MBEDTLS_PRIVATE(queue)[16]; /** The current partial block of data. */ - size_t MBEDTLS_PRIVATE(queue_len); /** The number of bytes stored in 'queue'. */ -} -mbedtls_poly1305_context; - -#else /* MBEDTLS_POLY1305_ALT */ -#include "poly1305_alt.h" -#endif /* MBEDTLS_POLY1305_ALT */ - -/** - * \brief This function initializes the specified Poly1305 context. - * - * It must be the first API called before using - * the context. - * - * It is usually followed by a call to - * \c mbedtls_poly1305_starts(), then one or more calls to - * \c mbedtls_poly1305_update(), then one call to - * \c mbedtls_poly1305_finish(), then finally - * \c mbedtls_poly1305_free(). - * - * \param ctx The Poly1305 context to initialize. This must - * not be \c NULL. - */ -void mbedtls_poly1305_init( mbedtls_poly1305_context *ctx ); - -/** - * \brief This function releases and clears the specified - * Poly1305 context. - * - * \param ctx The Poly1305 context to clear. This may be \c NULL, in which - * case this function is a no-op. If it is not \c NULL, it must - * point to an initialized Poly1305 context. - */ -void mbedtls_poly1305_free( mbedtls_poly1305_context *ctx ); - -/** - * \brief This function sets the one-time authentication key. - * - * \warning The key must be unique and unpredictable for each - * invocation of Poly1305. - * - * \param ctx The Poly1305 context to which the key should be bound. - * This must be initialized. - * \param key The buffer containing the \c 32 Byte (\c 256 Bit) key. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_poly1305_starts( mbedtls_poly1305_context *ctx, - const unsigned char key[32] ); - -/** - * \brief This functions feeds an input buffer into an ongoing - * Poly1305 computation. - * - * It is called between \c mbedtls_cipher_poly1305_starts() and - * \c mbedtls_cipher_poly1305_finish(). - * It can be called repeatedly to process a stream of data. - * - * \param ctx The Poly1305 context to use for the Poly1305 operation. - * This must be initialized and bound to a key. - * \param ilen The length of the input data in Bytes. - * Any value is accepted. - * \param input The buffer holding the input data. - * This pointer can be \c NULL if `ilen == 0`. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_poly1305_update( mbedtls_poly1305_context *ctx, - const unsigned char *input, - size_t ilen ); - -/** - * \brief This function generates the Poly1305 Message - * Authentication Code (MAC). - * - * \param ctx The Poly1305 context to use for the Poly1305 operation. - * This must be initialized and bound to a key. - * \param mac The buffer to where the MAC is written. This must - * be a writable buffer of length \c 16 Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_poly1305_finish( mbedtls_poly1305_context *ctx, - unsigned char mac[16] ); - -/** - * \brief This function calculates the Poly1305 MAC of the input - * buffer with the provided key. - * - * \warning The key must be unique and unpredictable for each - * invocation of Poly1305. - * - * \param key The buffer containing the \c 32 Byte (\c 256 Bit) key. - * \param ilen The length of the input data in Bytes. - * Any value is accepted. - * \param input The buffer holding the input data. - * This pointer can be \c NULL if `ilen == 0`. - * \param mac The buffer to where the MAC is written. This must be - * a writable buffer of length \c 16 Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_poly1305_mac( const unsigned char key[32], - const unsigned char *input, - size_t ilen, - unsigned char mac[16] ); - -#if defined(MBEDTLS_SELF_TEST) -/** - * \brief The Poly1305 checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_poly1305_self_test( int verbose ); -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_POLY1305_H */ diff --git a/windows/mbedtls/private_access.h b/windows/mbedtls/private_access.h deleted file mode 100644 index 98d3419f..00000000 --- a/windows/mbedtls/private_access.h +++ /dev/null @@ -1,32 +0,0 @@ - /** - * \file private_access.h - * - * \brief Macro wrapper for struct's memebrs. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_PRIVATE_ACCESS_H -#define MBEDTLS_PRIVATE_ACCESS_H - -#ifndef MBEDTLS_ALLOW_PRIVATE_ACCESS -#define MBEDTLS_PRIVATE(member) private_##member -#else -#define MBEDTLS_PRIVATE(member) member -#endif - -#endif /* MBEDTLS_PRIVATE_ACCESS_H */ diff --git a/windows/mbedtls/psa_util.h b/windows/mbedtls/psa_util.h deleted file mode 100644 index f6f2e580..00000000 --- a/windows/mbedtls/psa_util.h +++ /dev/null @@ -1,503 +0,0 @@ -/** - * \file psa_util.h - * - * \brief Utility functions for the use of the PSA Crypto library. - * - * \warning This function is not part of the public API and may - * change at any time. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#ifndef MBEDTLS_PSA_UTIL_H -#define MBEDTLS_PSA_UTIL_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#if defined(MBEDTLS_USE_PSA_CRYPTO) - -#include "psa/crypto.h" - -#include "mbedtls/ecp.h" -#include "mbedtls/md.h" -#include "mbedtls/pk.h" -#include "mbedtls/oid.h" -#include "mbedtls/error.h" - -#include - -/* Translations for symmetric crypto. */ - -static inline psa_key_type_t mbedtls_psa_translate_cipher_type( - mbedtls_cipher_type_t cipher ) -{ - switch( cipher ) - { - case MBEDTLS_CIPHER_AES_128_CCM: - case MBEDTLS_CIPHER_AES_192_CCM: - case MBEDTLS_CIPHER_AES_256_CCM: - case MBEDTLS_CIPHER_AES_128_GCM: - case MBEDTLS_CIPHER_AES_192_GCM: - case MBEDTLS_CIPHER_AES_256_GCM: - case MBEDTLS_CIPHER_AES_128_CBC: - case MBEDTLS_CIPHER_AES_192_CBC: - case MBEDTLS_CIPHER_AES_256_CBC: - return( PSA_KEY_TYPE_AES ); - - /* ARIA not yet supported in PSA. */ - /* case MBEDTLS_CIPHER_ARIA_128_CCM: - case MBEDTLS_CIPHER_ARIA_192_CCM: - case MBEDTLS_CIPHER_ARIA_256_CCM: - case MBEDTLS_CIPHER_ARIA_128_GCM: - case MBEDTLS_CIPHER_ARIA_192_GCM: - case MBEDTLS_CIPHER_ARIA_256_GCM: - case MBEDTLS_CIPHER_ARIA_128_CBC: - case MBEDTLS_CIPHER_ARIA_192_CBC: - case MBEDTLS_CIPHER_ARIA_256_CBC: - return( PSA_KEY_TYPE_ARIA ); */ - - default: - return( 0 ); - } -} - -static inline psa_algorithm_t mbedtls_psa_translate_cipher_mode( - mbedtls_cipher_mode_t mode, size_t taglen ) -{ - switch( mode ) - { - case MBEDTLS_MODE_ECB: - return( PSA_ALG_ECB_NO_PADDING ); - case MBEDTLS_MODE_GCM: - return( PSA_ALG_AEAD_WITH_SHORTENED_TAG( PSA_ALG_GCM, taglen ) ); - case MBEDTLS_MODE_CCM: - return( PSA_ALG_AEAD_WITH_SHORTENED_TAG( PSA_ALG_CCM, taglen ) ); - case MBEDTLS_MODE_CBC: - if( taglen == 0 ) - return( PSA_ALG_CBC_NO_PADDING ); - else - return( 0 ); - default: - return( 0 ); - } -} - -static inline psa_key_usage_t mbedtls_psa_translate_cipher_operation( - mbedtls_operation_t op ) -{ - switch( op ) - { - case MBEDTLS_ENCRYPT: - return( PSA_KEY_USAGE_ENCRYPT ); - case MBEDTLS_DECRYPT: - return( PSA_KEY_USAGE_DECRYPT ); - default: - return( 0 ); - } -} - -/* Translations for hashing. */ - -static inline psa_algorithm_t mbedtls_psa_translate_md( mbedtls_md_type_t md_alg ) -{ - switch( md_alg ) - { -#if defined(MBEDTLS_MD5_C) - case MBEDTLS_MD_MD5: - return( PSA_ALG_MD5 ); -#endif -#if defined(MBEDTLS_SHA1_C) - case MBEDTLS_MD_SHA1: - return( PSA_ALG_SHA_1 ); -#endif -#if defined(MBEDTLS_SHA224_C) - case MBEDTLS_MD_SHA224: - return( PSA_ALG_SHA_224 ); -#endif -#if defined(MBEDTLS_SHA256_C) - case MBEDTLS_MD_SHA256: - return( PSA_ALG_SHA_256 ); -#endif -#if defined(MBEDTLS_SHA384_C) - case MBEDTLS_MD_SHA384: - return( PSA_ALG_SHA_384 ); -#endif -#if defined(MBEDTLS_SHA512_C) - case MBEDTLS_MD_SHA512: - return( PSA_ALG_SHA_512 ); -#endif -#if defined(MBEDTLS_RIPEMD160_C) - case MBEDTLS_MD_RIPEMD160: - return( PSA_ALG_RIPEMD160 ); -#endif - case MBEDTLS_MD_NONE: - return( 0 ); - default: - return( 0 ); - } -} - -/* Translations for ECC. */ - -static inline int mbedtls_psa_get_ecc_oid_from_id( - psa_ecc_family_t curve, size_t bits, - char const **oid, size_t *oid_len ) -{ - switch( curve ) - { - case PSA_ECC_FAMILY_SECP_R1: - switch( bits ) - { -#if defined(MBEDTLS_ECP_DP_SECP192R1_ENABLED) - case 192: - *oid = MBEDTLS_OID_EC_GRP_SECP192R1; - *oid_len = MBEDTLS_OID_SIZE( MBEDTLS_OID_EC_GRP_SECP192R1 ); - return( 0 ); -#endif /* MBEDTLS_ECP_DP_SECP192R1_ENABLED */ -#if defined(MBEDTLS_ECP_DP_SECP224R1_ENABLED) - case 224: - *oid = MBEDTLS_OID_EC_GRP_SECP224R1; - *oid_len = MBEDTLS_OID_SIZE( MBEDTLS_OID_EC_GRP_SECP224R1 ); - return( 0 ); -#endif /* MBEDTLS_ECP_DP_SECP224R1_ENABLED */ -#if defined(MBEDTLS_ECP_DP_SECP256R1_ENABLED) - case 256: - *oid = MBEDTLS_OID_EC_GRP_SECP256R1; - *oid_len = MBEDTLS_OID_SIZE( MBEDTLS_OID_EC_GRP_SECP256R1 ); - return( 0 ); -#endif /* MBEDTLS_ECP_DP_SECP256R1_ENABLED */ -#if defined(MBEDTLS_ECP_DP_SECP384R1_ENABLED) - case 384: - *oid = MBEDTLS_OID_EC_GRP_SECP384R1; - *oid_len = MBEDTLS_OID_SIZE( MBEDTLS_OID_EC_GRP_SECP384R1 ); - return( 0 ); -#endif /* MBEDTLS_ECP_DP_SECP384R1_ENABLED */ -#if defined(MBEDTLS_ECP_DP_SECP521R1_ENABLED) - case 521: - *oid = MBEDTLS_OID_EC_GRP_SECP521R1; - *oid_len = MBEDTLS_OID_SIZE( MBEDTLS_OID_EC_GRP_SECP521R1 ); - return( 0 ); -#endif /* MBEDTLS_ECP_DP_SECP521R1_ENABLED */ - } - break; - case PSA_ECC_FAMILY_SECP_K1: - switch( bits ) - { -#if defined(MBEDTLS_ECP_DP_SECP192K1_ENABLED) - case 192: - *oid = MBEDTLS_OID_EC_GRP_SECP192K1; - *oid_len = MBEDTLS_OID_SIZE( MBEDTLS_OID_EC_GRP_SECP192K1 ); - return( 0 ); -#endif /* MBEDTLS_ECP_DP_SECP192K1_ENABLED */ -#if defined(MBEDTLS_ECP_DP_SECP224K1_ENABLED) - case 224: - *oid = MBEDTLS_OID_EC_GRP_SECP224K1; - *oid_len = MBEDTLS_OID_SIZE( MBEDTLS_OID_EC_GRP_SECP224K1 ); - return( 0 ); -#endif /* MBEDTLS_ECP_DP_SECP224K1_ENABLED */ -#if defined(MBEDTLS_ECP_DP_SECP256K1_ENABLED) - case 256: - *oid = MBEDTLS_OID_EC_GRP_SECP256K1; - *oid_len = MBEDTLS_OID_SIZE( MBEDTLS_OID_EC_GRP_SECP256K1 ); - return( 0 ); -#endif /* MBEDTLS_ECP_DP_SECP256K1_ENABLED */ - } - break; - case PSA_ECC_FAMILY_BRAINPOOL_P_R1: - switch( bits ) - { -#if defined(MBEDTLS_ECP_DP_BP256R1_ENABLED) - case 256: - *oid = MBEDTLS_OID_EC_GRP_BP256R1; - *oid_len = MBEDTLS_OID_SIZE( MBEDTLS_OID_EC_GRP_BP256R1 ); - return( 0 ); -#endif /* MBEDTLS_ECP_DP_BP256R1_ENABLED */ -#if defined(MBEDTLS_ECP_DP_BP384R1_ENABLED) - case 384: - *oid = MBEDTLS_OID_EC_GRP_BP384R1; - *oid_len = MBEDTLS_OID_SIZE( MBEDTLS_OID_EC_GRP_BP384R1 ); - return( 0 ); -#endif /* MBEDTLS_ECP_DP_BP384R1_ENABLED */ -#if defined(MBEDTLS_ECP_DP_BP512R1_ENABLED) - case 512: - *oid = MBEDTLS_OID_EC_GRP_BP512R1; - *oid_len = MBEDTLS_OID_SIZE( MBEDTLS_OID_EC_GRP_BP512R1 ); - return( 0 ); -#endif /* MBEDTLS_ECP_DP_BP512R1_ENABLED */ - } - break; - } - (void) oid; - (void) oid_len; - return( -1 ); -} - -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH 1 - -#if defined(MBEDTLS_ECP_DP_SECP192R1_ENABLED) -#if MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH < ( 2 * ( ( 192 + 7 ) / 8 ) + 1 ) -#undef MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH ( 2 * ( ( 192 + 7 ) / 8 ) + 1 ) -#endif -#endif /* MBEDTLS_ECP_DP_SECP192R1_ENABLED */ - -#if defined(MBEDTLS_ECP_DP_SECP224R1_ENABLED) -#if MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH < ( 2 * ( ( 224 + 7 ) / 8 ) + 1 ) -#undef MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH ( 2 * ( ( 224 + 7 ) / 8 ) + 1 ) -#endif -#endif /* MBEDTLS_ECP_DP_SECP224R1_ENABLED */ - -#if defined(MBEDTLS_ECP_DP_SECP256R1_ENABLED) -#if MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH < ( 2 * ( ( 256 + 7 ) / 8 ) + 1 ) -#undef MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH ( 2 * ( ( 256 + 7 ) / 8 ) + 1 ) -#endif -#endif /* MBEDTLS_ECP_DP_SECP256R1_ENABLED */ - -#if defined(MBEDTLS_ECP_DP_SECP384R1_ENABLED) -#if MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH < ( 2 * ( ( 384 + 7 ) / 8 ) + 1 ) -#undef MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH ( 2 * ( ( 384 + 7 ) / 8 ) + 1 ) -#endif -#endif /* MBEDTLS_ECP_DP_SECP384R1_ENABLED */ - -#if defined(MBEDTLS_ECP_DP_SECP521R1_ENABLED) -#if MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH < ( 2 * ( ( 521 + 7 ) / 8 ) + 1 ) -#undef MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH ( 2 * ( ( 521 + 7 ) / 8 ) + 1 ) -#endif -#endif /* MBEDTLS_ECP_DP_SECP521R1_ENABLED */ - -#if defined(MBEDTLS_ECP_DP_SECP192K1_ENABLED) -#if MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH < ( 2 * ( ( 192 + 7 ) / 8 ) + 1 ) -#undef MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH ( 2 * ( ( 192 + 7 ) / 8 ) + 1 ) -#endif -#endif /* MBEDTLS_ECP_DP_SECP192K1_ENABLED */ - -#if defined(MBEDTLS_ECP_DP_SECP224K1_ENABLED) -#if MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH < ( 2 * ( ( 224 + 7 ) / 8 ) + 1 ) -#undef MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH ( 2 * ( ( 224 + 7 ) / 8 ) + 1 ) -#endif -#endif /* MBEDTLS_ECP_DP_SECP224K1_ENABLED */ - -#if defined(MBEDTLS_ECP_DP_SECP256K1_ENABLED) -#if MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH < ( 2 * ( ( 256 + 7 ) / 8 ) + 1 ) -#undef MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH ( 2 * ( ( 256 + 7 ) / 8 ) + 1 ) -#endif -#endif /* MBEDTLS_ECP_DP_SECP256K1_ENABLED */ - -#if defined(MBEDTLS_ECP_DP_BP256R1_ENABLED) -#if MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH < ( 2 * ( ( 256 + 7 ) / 8 ) + 1 ) -#undef MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH ( 2 * ( ( 256 + 7 ) / 8 ) + 1 ) -#endif -#endif /* MBEDTLS_ECP_DP_BP256R1_ENABLED */ - -#if defined(MBEDTLS_ECP_DP_BP384R1_ENABLED) -#if MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH < ( 2 * ( ( 384 + 7 ) / 8 ) + 1 ) -#undef MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH ( 2 * ( ( 384 + 7 ) / 8 ) + 1 ) -#endif -#endif /* MBEDTLS_ECP_DP_BP384R1_ENABLED */ - -#if defined(MBEDTLS_ECP_DP_BP512R1_ENABLED) -#if MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH < ( 2 * ( ( 512 + 7 ) / 8 ) + 1 ) -#undef MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH -#define MBEDTLS_PSA_MAX_EC_PUBKEY_LENGTH ( 2 * ( ( 512 + 7 ) / 8 ) + 1 ) -#endif -#endif /* MBEDTLS_ECP_DP_BP512R1_ENABLED */ - - -/* Translations for PK layer */ - -static inline int mbedtls_psa_err_translate_pk( psa_status_t status ) -{ - switch( status ) - { - case PSA_SUCCESS: - return( 0 ); - case PSA_ERROR_NOT_SUPPORTED: - return( MBEDTLS_ERR_PK_FEATURE_UNAVAILABLE ); - case PSA_ERROR_INSUFFICIENT_MEMORY: - return( MBEDTLS_ERR_PK_ALLOC_FAILED ); - case PSA_ERROR_INSUFFICIENT_ENTROPY: - return( MBEDTLS_ERR_ECP_RANDOM_FAILED ); - case PSA_ERROR_BAD_STATE: - return( MBEDTLS_ERR_PK_BAD_INPUT_DATA ); - /* All other failures */ - case PSA_ERROR_COMMUNICATION_FAILURE: - case PSA_ERROR_HARDWARE_FAILURE: - case PSA_ERROR_CORRUPTION_DETECTED: - return( MBEDTLS_ERR_PLATFORM_HW_ACCEL_FAILED ); - default: /* We return the same as for the 'other failures', - * but list them separately nonetheless to indicate - * which failure conditions we have considered. */ - return( MBEDTLS_ERR_PLATFORM_HW_ACCEL_FAILED ); - } -} - -/* Translations for ECC */ - -/* This function transforms an ECC group identifier from - * https://www.iana.org/assignments/tls-parameters/tls-parameters.xhtml#tls-parameters-8 - * into a PSA ECC group identifier. */ -#if defined(MBEDTLS_ECP_C) -static inline psa_key_type_t mbedtls_psa_parse_tls_ecc_group( - uint16_t tls_ecc_grp_reg_id, size_t *bits ) -{ - const mbedtls_ecp_curve_info *curve_info = - mbedtls_ecp_curve_info_from_tls_id( tls_ecc_grp_reg_id ); - if( curve_info == NULL ) - return( 0 ); - return( PSA_KEY_TYPE_ECC_KEY_PAIR( - mbedtls_ecc_group_to_psa( curve_info->MBEDTLS_PRIVATE(grp_id), bits ) ) ); -} -#endif /* MBEDTLS_ECP_C */ - -/* This function takes a buffer holding an EC public key - * exported through psa_export_public_key(), and converts - * it into an ECPoint structure to be put into a ClientKeyExchange - * message in an ECDHE exchange. - * - * Both the present and the foreseeable future format of EC public keys - * used by PSA have the ECPoint structure contained in the exported key - * as a subbuffer, and the function merely selects this subbuffer instead - * of making a copy. - */ -static inline int mbedtls_psa_tls_psa_ec_to_ecpoint( unsigned char *src, - size_t srclen, - unsigned char **dst, - size_t *dstlen ) -{ - *dst = src; - *dstlen = srclen; - return( 0 ); -} - -/* This function takes a buffer holding an ECPoint structure - * (as contained in a TLS ServerKeyExchange message for ECDHE - * exchanges) and converts it into a format that the PSA key - * agreement API understands. - */ -static inline int mbedtls_psa_tls_ecpoint_to_psa_ec( unsigned char const *src, - size_t srclen, - unsigned char *dst, - size_t dstlen, - size_t *olen ) -{ - if( srclen > dstlen ) - return( MBEDTLS_ERR_ECP_BUFFER_TOO_SMALL ); - - memcpy( dst, src, srclen ); - *olen = srclen; - return( 0 ); -} - -#endif /* MBEDTLS_USE_PSA_CRYPTO */ - -/* Expose whatever RNG the PSA subsystem uses to applications using the - * mbedtls_xxx API. The declarations and definitions here need to be - * consistent with the implementation in library/psa_crypto_random_impl.h. - * See that file for implementation documentation. */ -#if defined(MBEDTLS_PSA_CRYPTO_C) - -/* The type of a `f_rng` random generator function that many library functions - * take. - * - * This type name is not part of the Mbed TLS stable API. It may be renamed - * or moved without warning. - */ -typedef int mbedtls_f_rng_t( void *p_rng, unsigned char *output, size_t output_size ); - -#if defined(MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG) - -/** The random generator function for the PSA subsystem. - * - * This function is suitable as the `f_rng` random generator function - * parameter of many `mbedtls_xxx` functions. Use #MBEDTLS_PSA_RANDOM_STATE - * to obtain the \p p_rng parameter. - * - * The implementation of this function depends on the configuration of the - * library. - * - * \note Depending on the configuration, this may be a function or - * a pointer to a function. - * - * \note This function may only be used if the PSA crypto subsystem is active. - * This means that you must call psa_crypto_init() before any call to - * this function, and you must not call this function after calling - * mbedtls_psa_crypto_free(). - * - * \param p_rng The random generator context. This must be - * #MBEDTLS_PSA_RANDOM_STATE. No other state is - * supported. - * \param output The buffer to fill. It must have room for - * \c output_size bytes. - * \param output_size The number of bytes to write to \p output. - * This function may fail if \p output_size is too - * large. It is guaranteed to accept any output size - * requested by Mbed TLS library functions. The - * maximum request size depends on the library - * configuration. - * - * \return \c 0 on success. - * \return An `MBEDTLS_ERR_ENTROPY_xxx`, - * `MBEDTLS_ERR_PLATFORM_xxx, - * `MBEDTLS_ERR_CTR_DRBG_xxx` or - * `MBEDTLS_ERR_HMAC_DRBG_xxx` on error. - */ -int mbedtls_psa_get_random( void *p_rng, - unsigned char *output, - size_t output_size ); - -/** The random generator state for the PSA subsystem. - * - * This macro expands to an expression which is suitable as the `p_rng` - * random generator state parameter of many `mbedtls_xxx` functions. - * It must be used in combination with the random generator function - * mbedtls_psa_get_random(). - * - * The implementation of this macro depends on the configuration of the - * library. Do not make any assumption on its nature. - */ -#define MBEDTLS_PSA_RANDOM_STATE NULL - -#else /* !defined(MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG) */ - -#if defined(MBEDTLS_CTR_DRBG_C) -#include "mbedtls/ctr_drbg.h" -typedef mbedtls_ctr_drbg_context mbedtls_psa_drbg_context_t; -static mbedtls_f_rng_t *const mbedtls_psa_get_random = mbedtls_ctr_drbg_random; -#elif defined(MBEDTLS_HMAC_DRBG_C) -#include "mbedtls/hmac_drbg.h" -typedef mbedtls_hmac_drbg_context mbedtls_psa_drbg_context_t; -static mbedtls_f_rng_t *const mbedtls_psa_get_random = mbedtls_hmac_drbg_random; -#endif -extern mbedtls_psa_drbg_context_t *const mbedtls_psa_random_state; - -#define MBEDTLS_PSA_RANDOM_STATE mbedtls_psa_random_state - -#endif /* !defined(MBEDTLS_PSA_CRYPTO_EXTERNAL_RNG) */ - -#endif /* MBEDTLS_PSA_CRYPTO_C */ - -#endif /* MBEDTLS_PSA_UTIL_H */ diff --git a/windows/mbedtls/ripemd160.h b/windows/mbedtls/ripemd160.h deleted file mode 100644 index 9132a83b..00000000 --- a/windows/mbedtls/ripemd160.h +++ /dev/null @@ -1,149 +0,0 @@ -/** - * \file ripemd160.h - * - * \brief RIPE MD-160 message digest - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_RIPEMD160_H -#define MBEDTLS_RIPEMD160_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_RIPEMD160_ALT) -// Regular implementation -// - -/** - * \brief RIPEMD-160 context structure - */ -typedef struct mbedtls_ripemd160_context -{ - uint32_t MBEDTLS_PRIVATE(total)[2]; /*!< number of bytes processed */ - uint32_t MBEDTLS_PRIVATE(state)[5]; /*!< intermediate digest state */ - unsigned char MBEDTLS_PRIVATE(buffer)[64]; /*!< data block being processed */ -} -mbedtls_ripemd160_context; - -#else /* MBEDTLS_RIPEMD160_ALT */ -#include "ripemd160_alt.h" -#endif /* MBEDTLS_RIPEMD160_ALT */ - -/** - * \brief Initialize RIPEMD-160 context - * - * \param ctx RIPEMD-160 context to be initialized - */ -void mbedtls_ripemd160_init( mbedtls_ripemd160_context *ctx ); - -/** - * \brief Clear RIPEMD-160 context - * - * \param ctx RIPEMD-160 context to be cleared - */ -void mbedtls_ripemd160_free( mbedtls_ripemd160_context *ctx ); - -/** - * \brief Clone (the state of) an RIPEMD-160 context - * - * \param dst The destination context - * \param src The context to be cloned - */ -void mbedtls_ripemd160_clone( mbedtls_ripemd160_context *dst, - const mbedtls_ripemd160_context *src ); - -/** - * \brief RIPEMD-160 context setup - * - * \param ctx context to be initialized - * - * \return 0 if successful - */ -int mbedtls_ripemd160_starts( mbedtls_ripemd160_context *ctx ); - -/** - * \brief RIPEMD-160 process buffer - * - * \param ctx RIPEMD-160 context - * \param input buffer holding the data - * \param ilen length of the input data - * - * \return 0 if successful - */ -int mbedtls_ripemd160_update( mbedtls_ripemd160_context *ctx, - const unsigned char *input, - size_t ilen ); - -/** - * \brief RIPEMD-160 final digest - * - * \param ctx RIPEMD-160 context - * \param output RIPEMD-160 checksum result - * - * \return 0 if successful - */ -int mbedtls_ripemd160_finish( mbedtls_ripemd160_context *ctx, - unsigned char output[20] ); - -/** - * \brief RIPEMD-160 process data block (internal use only) - * - * \param ctx RIPEMD-160 context - * \param data buffer holding one block of data - * - * \return 0 if successful - */ -int mbedtls_internal_ripemd160_process( mbedtls_ripemd160_context *ctx, - const unsigned char data[64] ); - -/** - * \brief Output = RIPEMD-160( input buffer ) - * - * \param input buffer holding the data - * \param ilen length of the input data - * \param output RIPEMD-160 checksum result - * - * \return 0 if successful - */ -int mbedtls_ripemd160( const unsigned char *input, - size_t ilen, - unsigned char output[20] ); - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief Checkup routine - * - * \return 0 if successful, or 1 if the test failed - */ -int mbedtls_ripemd160_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* mbedtls_ripemd160.h */ diff --git a/windows/mbedtls/rsa.h b/windows/mbedtls/rsa.h deleted file mode 100644 index cffbe3b5..00000000 --- a/windows/mbedtls/rsa.h +++ /dev/null @@ -1,1119 +0,0 @@ -/** - * \file rsa.h - * - * \brief This file provides an API for the RSA public-key cryptosystem. - * - * The RSA public-key cryptosystem is defined in Public-Key - * Cryptography Standards (PKCS) #1 v1.5: RSA Encryption - * and Public-Key Cryptography Standards (PKCS) #1 v2.1: - * RSA Cryptography Specifications. - * - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_RSA_H -#define MBEDTLS_RSA_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/bignum.h" -#include "mbedtls/md.h" - -#if defined(MBEDTLS_THREADING_C) -#include "mbedtls/threading.h" -#endif - -/* - * RSA Error codes - */ -#define MBEDTLS_ERR_RSA_BAD_INPUT_DATA -0x4080 /**< Bad input parameters to function. */ -#define MBEDTLS_ERR_RSA_INVALID_PADDING -0x4100 /**< Input data contains invalid padding and is rejected. */ -#define MBEDTLS_ERR_RSA_KEY_GEN_FAILED -0x4180 /**< Something failed during generation of a key. */ -#define MBEDTLS_ERR_RSA_KEY_CHECK_FAILED -0x4200 /**< Key failed to pass the validity check of the library. */ -#define MBEDTLS_ERR_RSA_PUBLIC_FAILED -0x4280 /**< The public key operation failed. */ -#define MBEDTLS_ERR_RSA_PRIVATE_FAILED -0x4300 /**< The private key operation failed. */ -#define MBEDTLS_ERR_RSA_VERIFY_FAILED -0x4380 /**< The PKCS#1 verification failed. */ -#define MBEDTLS_ERR_RSA_OUTPUT_TOO_LARGE -0x4400 /**< The output buffer for decryption is not large enough. */ -#define MBEDTLS_ERR_RSA_RNG_FAILED -0x4480 /**< The random generator failed to generate non-zeros. */ - -/* - * RSA constants - */ - -#define MBEDTLS_RSA_PKCS_V15 0 /**< Use PKCS#1 v1.5 encoding. */ -#define MBEDTLS_RSA_PKCS_V21 1 /**< Use PKCS#1 v2.1 encoding. */ - -#define MBEDTLS_RSA_SIGN 1 /**< Identifier for RSA signature operations. */ -#define MBEDTLS_RSA_CRYPT 2 /**< Identifier for RSA encryption and decryption operations. */ - -#define MBEDTLS_RSA_SALT_LEN_ANY -1 - -/* - * The above constants may be used even if the RSA module is compile out, - * eg for alternative (PKCS#11) RSA implemenations in the PK layers. - */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_RSA_ALT) -// Regular implementation -// - -/** - * \brief The RSA context structure. - */ -typedef struct mbedtls_rsa_context -{ - int MBEDTLS_PRIVATE(ver); /*!< Reserved for internal purposes. - * Do not set this field in application - * code. Its meaning might change without - * notice. */ - size_t MBEDTLS_PRIVATE(len); /*!< The size of \p N in Bytes. */ - - mbedtls_mpi MBEDTLS_PRIVATE(N); /*!< The public modulus. */ - mbedtls_mpi MBEDTLS_PRIVATE(E); /*!< The public exponent. */ - - mbedtls_mpi MBEDTLS_PRIVATE(D); /*!< The private exponent. */ - mbedtls_mpi MBEDTLS_PRIVATE(P); /*!< The first prime factor. */ - mbedtls_mpi MBEDTLS_PRIVATE(Q); /*!< The second prime factor. */ - - mbedtls_mpi MBEDTLS_PRIVATE(DP); /*!< D % (P - 1). */ - mbedtls_mpi MBEDTLS_PRIVATE(DQ); /*!< D % (Q - 1). */ - mbedtls_mpi MBEDTLS_PRIVATE(QP); /*!< 1 / (Q % P). */ - - mbedtls_mpi MBEDTLS_PRIVATE(RN); /*!< cached R^2 mod N. */ - - mbedtls_mpi MBEDTLS_PRIVATE(RP); /*!< cached R^2 mod P. */ - mbedtls_mpi MBEDTLS_PRIVATE(RQ); /*!< cached R^2 mod Q. */ - - mbedtls_mpi MBEDTLS_PRIVATE(Vi); /*!< The cached blinding value. */ - mbedtls_mpi MBEDTLS_PRIVATE(Vf); /*!< The cached un-blinding value. */ - - int MBEDTLS_PRIVATE(padding); /*!< Selects padding mode: - #MBEDTLS_RSA_PKCS_V15 for 1.5 padding and - #MBEDTLS_RSA_PKCS_V21 for OAEP or PSS. */ - int MBEDTLS_PRIVATE(hash_id); /*!< Hash identifier of mbedtls_md_type_t type, - as specified in md.h for use in the MGF - mask generating function used in the - EME-OAEP and EMSA-PSS encodings. */ -#if defined(MBEDTLS_THREADING_C) - /* Invariant: the mutex is initialized iff ver != 0. */ - mbedtls_threading_mutex_t MBEDTLS_PRIVATE(mutex); /*!< Thread-safety mutex. */ -#endif -} -mbedtls_rsa_context; - -#else /* MBEDTLS_RSA_ALT */ -#include "rsa_alt.h" -#endif /* MBEDTLS_RSA_ALT */ - -/** - * \brief This function initializes an RSA context. - * - * \note This function initializes the padding and the hash - * identifier to respectively #MBEDTLS_RSA_PKCS_V15 and - * #MBEDTLS_MD_NONE. See mbedtls_rsa_set_padding() for more - * information about those parameters. - * - * \param ctx The RSA context to initialize. This must not be \c NULL. - */ -void mbedtls_rsa_init( mbedtls_rsa_context *ctx ); - -/** - * \brief This function sets padding for an already initialized RSA - * context. - * - * \note Set padding to #MBEDTLS_RSA_PKCS_V21 for the RSAES-OAEP - * encryption scheme and the RSASSA-PSS signature scheme. - * - * \note The \p hash_id parameter is ignored when using - * #MBEDTLS_RSA_PKCS_V15 padding. - * - * \note The choice of padding mode is strictly enforced for private - * key operations, since there might be security concerns in - * mixing padding modes. For public key operations it is - * a default value, which can be overridden by calling specific - * \c mbedtls_rsa_rsaes_xxx or \c mbedtls_rsa_rsassa_xxx - * functions. - * - * \note The hash selected in \p hash_id is always used for OEAP - * encryption. For PSS signatures, it is always used for - * making signatures, but can be overridden for verifying them. - * If set to #MBEDTLS_MD_NONE, it is always overridden. - * - * \param ctx The initialized RSA context to be configured. - * \param padding The padding mode to use. This must be either - * #MBEDTLS_RSA_PKCS_V15 or #MBEDTLS_RSA_PKCS_V21. - * \param hash_id The hash identifier for PSS or OAEP, if \p padding is - * #MBEDTLS_RSA_PKCS_V21. #MBEDTLS_MD_NONE is accepted by this - * function but may be not suitable for some operations. - * Ignored if \p padding is #MBEDTLS_RSA_PKCS_V15. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_RSA_INVALID_PADDING failure: - * \p padding or \p hash_id is invalid. - */ -int mbedtls_rsa_set_padding( mbedtls_rsa_context *ctx, int padding, - mbedtls_md_type_t hash_id ); - -/** - * \brief This function imports a set of core parameters into an - * RSA context. - * - * \note This function can be called multiple times for successive - * imports, if the parameters are not simultaneously present. - * - * Any sequence of calls to this function should be followed - * by a call to mbedtls_rsa_complete(), which checks and - * completes the provided information to a ready-for-use - * public or private RSA key. - * - * \note See mbedtls_rsa_complete() for more information on which - * parameters are necessary to set up a private or public - * RSA key. - * - * \note The imported parameters are copied and need not be preserved - * for the lifetime of the RSA context being set up. - * - * \param ctx The initialized RSA context to store the parameters in. - * \param N The RSA modulus. This may be \c NULL. - * \param P The first prime factor of \p N. This may be \c NULL. - * \param Q The second prime factor of \p N. This may be \c NULL. - * \param D The private exponent. This may be \c NULL. - * \param E The public exponent. This may be \c NULL. - * - * \return \c 0 on success. - * \return A non-zero error code on failure. - */ -int mbedtls_rsa_import( mbedtls_rsa_context *ctx, - const mbedtls_mpi *N, - const mbedtls_mpi *P, const mbedtls_mpi *Q, - const mbedtls_mpi *D, const mbedtls_mpi *E ); - -/** - * \brief This function imports core RSA parameters, in raw big-endian - * binary format, into an RSA context. - * - * \note This function can be called multiple times for successive - * imports, if the parameters are not simultaneously present. - * - * Any sequence of calls to this function should be followed - * by a call to mbedtls_rsa_complete(), which checks and - * completes the provided information to a ready-for-use - * public or private RSA key. - * - * \note See mbedtls_rsa_complete() for more information on which - * parameters are necessary to set up a private or public - * RSA key. - * - * \note The imported parameters are copied and need not be preserved - * for the lifetime of the RSA context being set up. - * - * \param ctx The initialized RSA context to store the parameters in. - * \param N The RSA modulus. This may be \c NULL. - * \param N_len The Byte length of \p N; it is ignored if \p N == NULL. - * \param P The first prime factor of \p N. This may be \c NULL. - * \param P_len The Byte length of \p P; it ns ignored if \p P == NULL. - * \param Q The second prime factor of \p N. This may be \c NULL. - * \param Q_len The Byte length of \p Q; it is ignored if \p Q == NULL. - * \param D The private exponent. This may be \c NULL. - * \param D_len The Byte length of \p D; it is ignored if \p D == NULL. - * \param E The public exponent. This may be \c NULL. - * \param E_len The Byte length of \p E; it is ignored if \p E == NULL. - * - * \return \c 0 on success. - * \return A non-zero error code on failure. - */ -int mbedtls_rsa_import_raw( mbedtls_rsa_context *ctx, - unsigned char const *N, size_t N_len, - unsigned char const *P, size_t P_len, - unsigned char const *Q, size_t Q_len, - unsigned char const *D, size_t D_len, - unsigned char const *E, size_t E_len ); - -/** - * \brief This function completes an RSA context from - * a set of imported core parameters. - * - * To setup an RSA public key, precisely \p N and \p E - * must have been imported. - * - * To setup an RSA private key, sufficient information must - * be present for the other parameters to be derivable. - * - * The default implementation supports the following: - *
  • Derive \p P, \p Q from \p N, \p D, \p E.
  • - *
  • Derive \p N, \p D from \p P, \p Q, \p E.
- * Alternative implementations need not support these. - * - * If this function runs successfully, it guarantees that - * the RSA context can be used for RSA operations without - * the risk of failure or crash. - * - * \warning This function need not perform consistency checks - * for the imported parameters. In particular, parameters that - * are not needed by the implementation might be silently - * discarded and left unchecked. To check the consistency - * of the key material, see mbedtls_rsa_check_privkey(). - * - * \param ctx The initialized RSA context holding imported parameters. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_RSA_BAD_INPUT_DATA if the attempted derivations - * failed. - * - */ -int mbedtls_rsa_complete( mbedtls_rsa_context *ctx ); - -/** - * \brief This function exports the core parameters of an RSA key. - * - * If this function runs successfully, the non-NULL buffers - * pointed to by \p N, \p P, \p Q, \p D, and \p E are fully - * written, with additional unused space filled leading by - * zero Bytes. - * - * Possible reasons for returning - * #MBEDTLS_ERR_PLATFORM_FEATURE_UNSUPPORTED:
    - *
  • An alternative RSA implementation is in use, which - * stores the key externally, and either cannot or should - * not export it into RAM.
  • - *
  • A SW or HW implementation might not support a certain - * deduction. For example, \p P, \p Q from \p N, \p D, - * and \p E if the former are not part of the - * implementation.
- * - * If the function fails due to an unsupported operation, - * the RSA context stays intact and remains usable. - * - * \param ctx The initialized RSA context. - * \param N The MPI to hold the RSA modulus. - * This may be \c NULL if this field need not be exported. - * \param P The MPI to hold the first prime factor of \p N. - * This may be \c NULL if this field need not be exported. - * \param Q The MPI to hold the second prime factor of \p N. - * This may be \c NULL if this field need not be exported. - * \param D The MPI to hold the private exponent. - * This may be \c NULL if this field need not be exported. - * \param E The MPI to hold the public exponent. - * This may be \c NULL if this field need not be exported. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_PLATFORM_FEATURE_UNSUPPORTED if exporting the - * requested parameters cannot be done due to missing - * functionality or because of security policies. - * \return A non-zero return code on any other failure. - * - */ -int mbedtls_rsa_export( const mbedtls_rsa_context *ctx, - mbedtls_mpi *N, mbedtls_mpi *P, mbedtls_mpi *Q, - mbedtls_mpi *D, mbedtls_mpi *E ); - -/** - * \brief This function exports core parameters of an RSA key - * in raw big-endian binary format. - * - * If this function runs successfully, the non-NULL buffers - * pointed to by \p N, \p P, \p Q, \p D, and \p E are fully - * written, with additional unused space filled leading by - * zero Bytes. - * - * Possible reasons for returning - * #MBEDTLS_ERR_PLATFORM_FEATURE_UNSUPPORTED:
    - *
  • An alternative RSA implementation is in use, which - * stores the key externally, and either cannot or should - * not export it into RAM.
  • - *
  • A SW or HW implementation might not support a certain - * deduction. For example, \p P, \p Q from \p N, \p D, - * and \p E if the former are not part of the - * implementation.
- * If the function fails due to an unsupported operation, - * the RSA context stays intact and remains usable. - * - * \note The length parameters are ignored if the corresponding - * buffer pointers are NULL. - * - * \param ctx The initialized RSA context. - * \param N The Byte array to store the RSA modulus, - * or \c NULL if this field need not be exported. - * \param N_len The size of the buffer for the modulus. - * \param P The Byte array to hold the first prime factor of \p N, - * or \c NULL if this field need not be exported. - * \param P_len The size of the buffer for the first prime factor. - * \param Q The Byte array to hold the second prime factor of \p N, - * or \c NULL if this field need not be exported. - * \param Q_len The size of the buffer for the second prime factor. - * \param D The Byte array to hold the private exponent, - * or \c NULL if this field need not be exported. - * \param D_len The size of the buffer for the private exponent. - * \param E The Byte array to hold the public exponent, - * or \c NULL if this field need not be exported. - * \param E_len The size of the buffer for the public exponent. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_PLATFORM_FEATURE_UNSUPPORTED if exporting the - * requested parameters cannot be done due to missing - * functionality or because of security policies. - * \return A non-zero return code on any other failure. - */ -int mbedtls_rsa_export_raw( const mbedtls_rsa_context *ctx, - unsigned char *N, size_t N_len, - unsigned char *P, size_t P_len, - unsigned char *Q, size_t Q_len, - unsigned char *D, size_t D_len, - unsigned char *E, size_t E_len ); - -/** - * \brief This function exports CRT parameters of a private RSA key. - * - * \note Alternative RSA implementations not using CRT-parameters - * internally can implement this function based on - * mbedtls_rsa_deduce_opt(). - * - * \param ctx The initialized RSA context. - * \param DP The MPI to hold \c D modulo `P-1`, - * or \c NULL if it need not be exported. - * \param DQ The MPI to hold \c D modulo `Q-1`, - * or \c NULL if it need not be exported. - * \param QP The MPI to hold modular inverse of \c Q modulo \c P, - * or \c NULL if it need not be exported. - * - * \return \c 0 on success. - * \return A non-zero error code on failure. - * - */ -int mbedtls_rsa_export_crt( const mbedtls_rsa_context *ctx, - mbedtls_mpi *DP, mbedtls_mpi *DQ, mbedtls_mpi *QP ); - -/** - * \brief This function retrieves the length of RSA modulus in Bytes. - * - * \param ctx The initialized RSA context. - * - * \return The length of the RSA modulus in Bytes. - * - */ -size_t mbedtls_rsa_get_len( const mbedtls_rsa_context *ctx ); - -/** - * \brief This function generates an RSA keypair. - * - * \note mbedtls_rsa_init() must be called before this function, - * to set up the RSA context. - * - * \param ctx The initialized RSA context used to hold the key. - * \param f_rng The RNG function to be used for key generation. - * This is mandatory and must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. - * This may be \c NULL if \p f_rng doesn't need a context. - * \param nbits The size of the public key in bits. - * \param exponent The public exponent to use. For example, \c 65537. - * This must be odd and greater than \c 1. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_gen_key( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - unsigned int nbits, int exponent ); - -/** - * \brief This function checks if a context contains at least an RSA - * public key. - * - * If the function runs successfully, it is guaranteed that - * enough information is present to perform an RSA public key - * operation using mbedtls_rsa_public(). - * - * \param ctx The initialized RSA context to check. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - * - */ -int mbedtls_rsa_check_pubkey( const mbedtls_rsa_context *ctx ); - -/** - * \brief This function checks if a context contains an RSA private key - * and perform basic consistency checks. - * - * \note The consistency checks performed by this function not only - * ensure that mbedtls_rsa_private() can be called successfully - * on the given context, but that the various parameters are - * mutually consistent with high probability, in the sense that - * mbedtls_rsa_public() and mbedtls_rsa_private() are inverses. - * - * \warning This function should catch accidental misconfigurations - * like swapping of parameters, but it cannot establish full - * trust in neither the quality nor the consistency of the key - * material that was used to setup the given RSA context: - *
  • Consistency: Imported parameters that are irrelevant - * for the implementation might be silently dropped. If dropped, - * the current function does not have access to them, - * and therefore cannot check them. See mbedtls_rsa_complete(). - * If you want to check the consistency of the entire - * content of an PKCS1-encoded RSA private key, for example, you - * should use mbedtls_rsa_validate_params() before setting - * up the RSA context. - * Additionally, if the implementation performs empirical checks, - * these checks substantiate but do not guarantee consistency.
  • - *
  • Quality: This function is not expected to perform - * extended quality assessments like checking that the prime - * factors are safe. Additionally, it is the responsibility of the - * user to ensure the trustworthiness of the source of his RSA - * parameters, which goes beyond what is effectively checkable - * by the library.
- * - * \param ctx The initialized RSA context to check. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_check_privkey( const mbedtls_rsa_context *ctx ); - -/** - * \brief This function checks a public-private RSA key pair. - * - * It checks each of the contexts, and makes sure they match. - * - * \param pub The initialized RSA context holding the public key. - * \param prv The initialized RSA context holding the private key. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_check_pub_priv( const mbedtls_rsa_context *pub, - const mbedtls_rsa_context *prv ); - -/** - * \brief This function performs an RSA public key operation. - * - * \param ctx The initialized RSA context to use. - * \param input The input buffer. This must be a readable buffer - * of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * \param output The output buffer. This must be a writable buffer - * of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * - * \note This function does not handle message padding. - * - * \note Make sure to set \p input[0] = 0 or ensure that - * input is smaller than \p N. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_public( mbedtls_rsa_context *ctx, - const unsigned char *input, - unsigned char *output ); - -/** - * \brief This function performs an RSA private key operation. - * - * \note Blinding is used if and only if a PRNG is provided. - * - * \note If blinding is used, both the base of exponentation - * and the exponent are blinded, providing protection - * against some side-channel attacks. - * - * \warning It is deprecated and a security risk to not provide - * a PRNG here and thereby prevent the use of blinding. - * Future versions of the library may enforce the presence - * of a PRNG. - * - * \param ctx The initialized RSA context to use. - * \param f_rng The RNG function, used for blinding. It is mandatory. - * \param p_rng The RNG context to pass to \p f_rng. This may be \c NULL - * if \p f_rng doesn't need a context. - * \param input The input buffer. This must be a readable buffer - * of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * \param output The output buffer. This must be a writable buffer - * of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - * - */ -int mbedtls_rsa_private( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - const unsigned char *input, - unsigned char *output ); - -/** - * \brief This function adds the message padding, then performs an RSA - * operation. - * - * It is the generic wrapper for performing a PKCS#1 encryption - * operation. - * - * \param ctx The initialized RSA context to use. - * \param f_rng The RNG to use. It is used for padding generation - * and it is mandatory. - * \param p_rng The RNG context to be passed to \p f_rng. May be - * \c NULL if \p f_rng doesn't need a context argument. - * \param ilen The length of the plaintext in Bytes. - * \param input The input data to encrypt. This must be a readable - * buffer of size \p ilen Bytes. It may be \c NULL if - * `ilen == 0`. - * \param output The output buffer. This must be a writable buffer - * of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_pkcs1_encrypt( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - size_t ilen, - const unsigned char *input, - unsigned char *output ); - -/** - * \brief This function performs a PKCS#1 v1.5 encryption operation - * (RSAES-PKCS1-v1_5-ENCRYPT). - * - * \param ctx The initialized RSA context to use. - * \param f_rng The RNG function to use. It is mandatory and used for - * padding generation. - * \param p_rng The RNG context to be passed to \p f_rng. This may - * be \c NULL if \p f_rng doesn't need a context argument. - * \param ilen The length of the plaintext in Bytes. - * \param input The input data to encrypt. This must be a readable - * buffer of size \p ilen Bytes. It may be \c NULL if - * `ilen == 0`. - * \param output The output buffer. This must be a writable buffer - * of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_rsaes_pkcs1_v15_encrypt( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - size_t ilen, - const unsigned char *input, - unsigned char *output ); - -/** - * \brief This function performs a PKCS#1 v2.1 OAEP encryption - * operation (RSAES-OAEP-ENCRYPT). - * - * \note The output buffer must be as large as the size - * of ctx->N. For example, 128 Bytes if RSA-1024 is used. - * - * \param ctx The initnialized RSA context to use. - * \param f_rng The RNG function to use. This is needed for padding - * generation and is mandatory. - * \param p_rng The RNG context to be passed to \p f_rng. This may - * be \c NULL if \p f_rng doesn't need a context argument. - * \param label The buffer holding the custom label to use. - * This must be a readable buffer of length \p label_len - * Bytes. It may be \c NULL if \p label_len is \c 0. - * \param label_len The length of the label in Bytes. - * \param ilen The length of the plaintext buffer \p input in Bytes. - * \param input The input data to encrypt. This must be a readable - * buffer of size \p ilen Bytes. It may be \c NULL if - * `ilen == 0`. - * \param output The output buffer. This must be a writable buffer - * of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_rsaes_oaep_encrypt( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - const unsigned char *label, size_t label_len, - size_t ilen, - const unsigned char *input, - unsigned char *output ); - -/** - * \brief This function performs an RSA operation, then removes the - * message padding. - * - * It is the generic wrapper for performing a PKCS#1 decryption - * operation. - * - * \note The output buffer length \c output_max_len should be - * as large as the size \p ctx->len of \p ctx->N (for example, - * 128 Bytes if RSA-1024 is used) to be able to hold an - * arbitrary decrypted message. If it is not large enough to - * hold the decryption of the particular ciphertext provided, - * the function returns \c MBEDTLS_ERR_RSA_OUTPUT_TOO_LARGE. - * - * \param ctx The initialized RSA context to use. - * \param f_rng The RNG function. This is used for blinding and is - * mandatory; see mbedtls_rsa_private() for more. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL if \p f_rng doesn't need a context. - * \param olen The address at which to store the length of - * the plaintext. This must not be \c NULL. - * \param input The ciphertext buffer. This must be a readable buffer - * of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * \param output The buffer used to hold the plaintext. This must - * be a writable buffer of length \p output_max_len Bytes. - * \param output_max_len The length in Bytes of the output buffer \p output. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_pkcs1_decrypt( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - size_t *olen, - const unsigned char *input, - unsigned char *output, - size_t output_max_len ); - -/** - * \brief This function performs a PKCS#1 v1.5 decryption - * operation (RSAES-PKCS1-v1_5-DECRYPT). - * - * \note The output buffer length \c output_max_len should be - * as large as the size \p ctx->len of \p ctx->N, for example, - * 128 Bytes if RSA-1024 is used, to be able to hold an - * arbitrary decrypted message. If it is not large enough to - * hold the decryption of the particular ciphertext provided, - * the function returns #MBEDTLS_ERR_RSA_OUTPUT_TOO_LARGE. - * - * \param ctx The initialized RSA context to use. - * \param f_rng The RNG function. This is used for blinding and is - * mandatory; see mbedtls_rsa_private() for more. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL if \p f_rng doesn't need a context. - * \param olen The address at which to store the length of - * the plaintext. This must not be \c NULL. - * \param input The ciphertext buffer. This must be a readable buffer - * of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * \param output The buffer used to hold the plaintext. This must - * be a writable buffer of length \p output_max_len Bytes. - * \param output_max_len The length in Bytes of the output buffer \p output. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - * - */ -int mbedtls_rsa_rsaes_pkcs1_v15_decrypt( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - size_t *olen, - const unsigned char *input, - unsigned char *output, - size_t output_max_len ); - -/** - * \brief This function performs a PKCS#1 v2.1 OAEP decryption - * operation (RSAES-OAEP-DECRYPT). - * - * \note The output buffer length \c output_max_len should be - * as large as the size \p ctx->len of \p ctx->N, for - * example, 128 Bytes if RSA-1024 is used, to be able to - * hold an arbitrary decrypted message. If it is not - * large enough to hold the decryption of the particular - * ciphertext provided, the function returns - * #MBEDTLS_ERR_RSA_OUTPUT_TOO_LARGE. - * - * \param ctx The initialized RSA context to use. - * \param f_rng The RNG function. This is used for blinding and is - * mandatory. - * \param p_rng The RNG context to be passed to \p f_rng. This may be - * \c NULL if \p f_rng doesn't need a context. - * \param label The buffer holding the custom label to use. - * This must be a readable buffer of length \p label_len - * Bytes. It may be \c NULL if \p label_len is \c 0. - * \param label_len The length of the label in Bytes. - * \param olen The address at which to store the length of - * the plaintext. This must not be \c NULL. - * \param input The ciphertext buffer. This must be a readable buffer - * of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * \param output The buffer used to hold the plaintext. This must - * be a writable buffer of length \p output_max_len Bytes. - * \param output_max_len The length in Bytes of the output buffer \p output. - * - * \return \c 0 on success. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_rsaes_oaep_decrypt( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - const unsigned char *label, size_t label_len, - size_t *olen, - const unsigned char *input, - unsigned char *output, - size_t output_max_len ); - -/** - * \brief This function performs a private RSA operation to sign - * a message digest using PKCS#1. - * - * It is the generic wrapper for performing a PKCS#1 - * signature. - * - * \note The \p sig buffer must be as large as the size - * of \p ctx->N. For example, 128 Bytes if RSA-1024 is used. - * - * \note For PKCS#1 v2.1 encoding, see comments on - * mbedtls_rsa_rsassa_pss_sign() for details on - * \p md_alg and \p hash_id. - * - * \param ctx The initialized RSA context to use. - * \param f_rng The RNG function to use. This is mandatory and - * must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be \c NULL - * if \p f_rng doesn't need a context argument. - * \param md_alg The message-digest algorithm used to hash the original data. - * Use #MBEDTLS_MD_NONE for signing raw data. - * \param hashlen The length of the message digest or raw data in Bytes. - * If \p md_alg is not #MBEDTLS_MD_NONE, this must match the - * output length of the corresponding hash algorithm. - * \param hash The buffer holding the message digest or raw data. - * This must be a readable buffer of at least \p hashlen Bytes. - * \param sig The buffer to hold the signature. This must be a writable - * buffer of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. A buffer length of - * #MBEDTLS_MPI_MAX_SIZE is always safe. - * - * \return \c 0 if the signing operation was successful. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_pkcs1_sign( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - mbedtls_md_type_t md_alg, - unsigned int hashlen, - const unsigned char *hash, - unsigned char *sig ); - -/** - * \brief This function performs a PKCS#1 v1.5 signature - * operation (RSASSA-PKCS1-v1_5-SIGN). - * - * \param ctx The initialized RSA context to use. - * \param f_rng The RNG function. This is used for blinding and is - * mandatory; see mbedtls_rsa_private() for more. - * \param p_rng The RNG context to be passed to \p f_rng. This may be \c NULL - * if \p f_rng doesn't need a context argument. - * \param md_alg The message-digest algorithm used to hash the original data. - * Use #MBEDTLS_MD_NONE for signing raw data. - * \param hashlen The length of the message digest or raw data in Bytes. - * If \p md_alg is not #MBEDTLS_MD_NONE, this must match the - * output length of the corresponding hash algorithm. - * \param hash The buffer holding the message digest or raw data. - * This must be a readable buffer of at least \p hashlen Bytes. - * \param sig The buffer to hold the signature. This must be a writable - * buffer of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. A buffer length of - * #MBEDTLS_MPI_MAX_SIZE is always safe. - * - * \return \c 0 if the signing operation was successful. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_rsassa_pkcs1_v15_sign( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - mbedtls_md_type_t md_alg, - unsigned int hashlen, - const unsigned char *hash, - unsigned char *sig ); - -/** - * \brief This function performs a PKCS#1 v2.1 PSS signature - * operation (RSASSA-PSS-SIGN). - * - * \note The \c hash_id set in \p ctx by calling - * mbedtls_rsa_set_padding() selects the hash used for the - * encoding operation and for the mask generation function - * (MGF1). For more details on the encoding operation and the - * mask generation function, consult RFC-3447: Public-Key - * Cryptography Standards (PKCS) #1 v2.1: RSA Cryptography - * Specifications. - * - * \note This function enforces that the provided salt length complies - * with FIPS 186-4 §5.5 (e) and RFC 8017 (PKCS#1 v2.2) §9.1.1 - * step 3. The constraint is that the hash length plus the salt - * length plus 2 bytes must be at most the key length. If this - * constraint is not met, this function returns - * #MBEDTLS_ERR_RSA_BAD_INPUT_DATA. - * - * \param ctx The initialized RSA context to use. - * \param f_rng The RNG function. It is mandatory and must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be \c NULL - * if \p f_rng doesn't need a context argument. - * \param md_alg The message-digest algorithm used to hash the original data. - * Use #MBEDTLS_MD_NONE for signing raw data. - * \param hashlen The length of the message digest or raw data in Bytes. - * If \p md_alg is not #MBEDTLS_MD_NONE, this must match the - * output length of the corresponding hash algorithm. - * \param hash The buffer holding the message digest or raw data. - * This must be a readable buffer of at least \p hashlen Bytes. - * \param saltlen The length of the salt that should be used. - * If passed #MBEDTLS_RSA_SALT_LEN_ANY, the function will use - * the largest possible salt length up to the hash length, - * which is the largest permitted by some standards including - * FIPS 186-4 §5.5. - * \param sig The buffer to hold the signature. This must be a writable - * buffer of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. A buffer length of - * #MBEDTLS_MPI_MAX_SIZE is always safe. - * - * \return \c 0 if the signing operation was successful. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_rsassa_pss_sign_ext( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - mbedtls_md_type_t md_alg, - unsigned int hashlen, - const unsigned char *hash, - int saltlen, - unsigned char *sig ); - -/** - * \brief This function performs a PKCS#1 v2.1 PSS signature - * operation (RSASSA-PSS-SIGN). - * - * \note The \c hash_id set in \p ctx by calling - * mbedtls_rsa_set_padding() selects the hash used for the - * encoding operation and for the mask generation function - * (MGF1). For more details on the encoding operation and the - * mask generation function, consult RFC-3447: Public-Key - * Cryptography Standards (PKCS) #1 v2.1: RSA Cryptography - * Specifications. - * - * \note This function always uses the maximum possible salt size, - * up to the length of the payload hash. This choice of salt - * size complies with FIPS 186-4 §5.5 (e) and RFC 8017 (PKCS#1 - * v2.2) §9.1.1 step 3. Furthermore this function enforces a - * minimum salt size which is the hash size minus 2 bytes. If - * this minimum size is too large given the key size (the salt - * size, plus the hash size, plus 2 bytes must be no more than - * the key size in bytes), this function returns - * #MBEDTLS_ERR_RSA_BAD_INPUT_DATA. - * - * \param ctx The initialized RSA context to use. - * \param f_rng The RNG function. It is mandatory and must not be \c NULL. - * \param p_rng The RNG context to be passed to \p f_rng. This may be \c NULL - * if \p f_rng doesn't need a context argument. - * \param md_alg The message-digest algorithm used to hash the original data. - * Use #MBEDTLS_MD_NONE for signing raw data. - * \param hashlen The length of the message digest or raw data in Bytes. - * If \p md_alg is not #MBEDTLS_MD_NONE, this must match the - * output length of the corresponding hash algorithm. - * \param hash The buffer holding the message digest or raw data. - * This must be a readable buffer of at least \p hashlen Bytes. - * \param sig The buffer to hold the signature. This must be a writable - * buffer of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. A buffer length of - * #MBEDTLS_MPI_MAX_SIZE is always safe. - * - * \return \c 0 if the signing operation was successful. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_rsassa_pss_sign( mbedtls_rsa_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng, - mbedtls_md_type_t md_alg, - unsigned int hashlen, - const unsigned char *hash, - unsigned char *sig ); - -/** - * \brief This function performs a public RSA operation and checks - * the message digest. - * - * This is the generic wrapper for performing a PKCS#1 - * verification. - * - * \note For PKCS#1 v2.1 encoding, see comments on - * mbedtls_rsa_rsassa_pss_verify() about \p md_alg and - * \p hash_id. - * - * \param ctx The initialized RSA public key context to use. - * \param md_alg The message-digest algorithm used to hash the original data. - * Use #MBEDTLS_MD_NONE for signing raw data. - * \param hashlen The length of the message digest or raw data in Bytes. - * If \p md_alg is not #MBEDTLS_MD_NONE, this must match the - * output length of the corresponding hash algorithm. - * \param hash The buffer holding the message digest or raw data. - * This must be a readable buffer of at least \p hashlen Bytes. - * \param sig The buffer holding the signature. This must be a readable - * buffer of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * - * \return \c 0 if the verify operation was successful. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_pkcs1_verify( mbedtls_rsa_context *ctx, - mbedtls_md_type_t md_alg, - unsigned int hashlen, - const unsigned char *hash, - const unsigned char *sig ); - -/** - * \brief This function performs a PKCS#1 v1.5 verification - * operation (RSASSA-PKCS1-v1_5-VERIFY). - * - * \param ctx The initialized RSA public key context to use. - * \param md_alg The message-digest algorithm used to hash the original data. - * Use #MBEDTLS_MD_NONE for signing raw data. - * \param hashlen The length of the message digest or raw data in Bytes. - * If \p md_alg is not #MBEDTLS_MD_NONE, this must match the - * output length of the corresponding hash algorithm. - * \param hash The buffer holding the message digest or raw data. - * This must be a readable buffer of at least \p hashlen Bytes. - * \param sig The buffer holding the signature. This must be a readable - * buffer of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * - * \return \c 0 if the verify operation was successful. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_rsassa_pkcs1_v15_verify( mbedtls_rsa_context *ctx, - mbedtls_md_type_t md_alg, - unsigned int hashlen, - const unsigned char *hash, - const unsigned char *sig ); - -/** - * \brief This function performs a PKCS#1 v2.1 PSS verification - * operation (RSASSA-PSS-VERIFY). - * - * \note The \c hash_id set in \p ctx by calling - * mbedtls_rsa_set_padding() selects the hash used for the - * encoding operation and for the mask generation function - * (MGF1). For more details on the encoding operation and the - * mask generation function, consult RFC-3447: Public-Key - * Cryptography Standards (PKCS) #1 v2.1: RSA Cryptography - * Specifications. If the \c hash_id set in \p ctx by - * mbedtls_rsa_set_padding() is #MBEDTLS_MD_NONE, the \p md_alg - * parameter is used. - * - * \param ctx The initialized RSA public key context to use. - * \param md_alg The message-digest algorithm used to hash the original data. - * Use #MBEDTLS_MD_NONE for signing raw data. - * \param hashlen The length of the message digest or raw data in Bytes. - * If \p md_alg is not #MBEDTLS_MD_NONE, this must match the - * output length of the corresponding hash algorithm. - * \param hash The buffer holding the message digest or raw data. - * This must be a readable buffer of at least \p hashlen Bytes. - * \param sig The buffer holding the signature. This must be a readable - * buffer of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * - * \return \c 0 if the verify operation was successful. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_rsassa_pss_verify( mbedtls_rsa_context *ctx, - mbedtls_md_type_t md_alg, - unsigned int hashlen, - const unsigned char *hash, - const unsigned char *sig ); - -/** - * \brief This function performs a PKCS#1 v2.1 PSS verification - * operation (RSASSA-PSS-VERIFY). - * - * \note The \p sig buffer must be as large as the size - * of \p ctx->N. For example, 128 Bytes if RSA-1024 is used. - * - * \note The \c hash_id set in \p ctx by mbedtls_rsa_set_padding() is - * ignored. - * - * \param ctx The initialized RSA public key context to use. - * \param md_alg The message-digest algorithm used to hash the original data. - * Use #MBEDTLS_MD_NONE for signing raw data. - * \param hashlen The length of the message digest or raw data in Bytes. - * If \p md_alg is not #MBEDTLS_MD_NONE, this must match the - * output length of the corresponding hash algorithm. - * \param hash The buffer holding the message digest or raw data. - * This must be a readable buffer of at least \p hashlen Bytes. - * \param mgf1_hash_id The message digest algorithm used for the - * verification operation and the mask generation - * function (MGF1). For more details on the encoding - * operation and the mask generation function, consult - * RFC-3447: Public-Key Cryptography Standards - * (PKCS) #1 v2.1: RSA Cryptography - * Specifications. - * \param expected_salt_len The length of the salt used in padding. Use - * #MBEDTLS_RSA_SALT_LEN_ANY to accept any salt length. - * \param sig The buffer holding the signature. This must be a readable - * buffer of length \c ctx->len Bytes. For example, \c 256 Bytes - * for an 2048-bit RSA modulus. - * - * \return \c 0 if the verify operation was successful. - * \return An \c MBEDTLS_ERR_RSA_XXX error code on failure. - */ -int mbedtls_rsa_rsassa_pss_verify_ext( mbedtls_rsa_context *ctx, - mbedtls_md_type_t md_alg, - unsigned int hashlen, - const unsigned char *hash, - mbedtls_md_type_t mgf1_hash_id, - int expected_salt_len, - const unsigned char *sig ); - -/** - * \brief This function copies the components of an RSA context. - * - * \param dst The destination context. This must be initialized. - * \param src The source context. This must be initialized. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_MPI_ALLOC_FAILED on memory allocation failure. - */ -int mbedtls_rsa_copy( mbedtls_rsa_context *dst, const mbedtls_rsa_context *src ); - -/** - * \brief This function frees the components of an RSA key. - * - * \param ctx The RSA context to free. May be \c NULL, in which case - * this function is a no-op. If it is not \c NULL, it must - * point to an initialized RSA context. - */ -void mbedtls_rsa_free( mbedtls_rsa_context *ctx ); - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief The RSA checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_rsa_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* rsa.h */ diff --git a/windows/mbedtls/sha1.h b/windows/mbedtls/sha1.h deleted file mode 100644 index d2db67ca..00000000 --- a/windows/mbedtls/sha1.h +++ /dev/null @@ -1,231 +0,0 @@ -/** - * \file sha1.h - * - * \brief This file contains SHA-1 definitions and functions. - * - * The Secure Hash Algorithm 1 (SHA-1) cryptographic hash function is defined in - * FIPS 180-4: Secure Hash Standard (SHS). - * - * \warning SHA-1 is considered a weak message digest and its use constitutes - * a security risk. We recommend considering stronger message - * digests instead. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_SHA1_H -#define MBEDTLS_SHA1_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -#define MBEDTLS_ERR_SHA1_BAD_INPUT_DATA -0x0073 /**< SHA-1 input data was malformed. */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_SHA1_ALT) -// Regular implementation -// - -/** - * \brief The SHA-1 context structure. - * - * \warning SHA-1 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - */ -typedef struct mbedtls_sha1_context -{ - uint32_t MBEDTLS_PRIVATE(total)[2]; /*!< The number of Bytes processed. */ - uint32_t MBEDTLS_PRIVATE(state)[5]; /*!< The intermediate digest state. */ - unsigned char MBEDTLS_PRIVATE(buffer)[64]; /*!< The data block being processed. */ -} -mbedtls_sha1_context; - -#else /* MBEDTLS_SHA1_ALT */ -#include "sha1_alt.h" -#endif /* MBEDTLS_SHA1_ALT */ - -/** - * \brief This function initializes a SHA-1 context. - * - * \warning SHA-1 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - * \param ctx The SHA-1 context to initialize. - * This must not be \c NULL. - * - */ -void mbedtls_sha1_init( mbedtls_sha1_context *ctx ); - -/** - * \brief This function clears a SHA-1 context. - * - * \warning SHA-1 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - * \param ctx The SHA-1 context to clear. This may be \c NULL, - * in which case this function does nothing. If it is - * not \c NULL, it must point to an initialized - * SHA-1 context. - * - */ -void mbedtls_sha1_free( mbedtls_sha1_context *ctx ); - -/** - * \brief This function clones the state of a SHA-1 context. - * - * \warning SHA-1 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - * \param dst The SHA-1 context to clone to. This must be initialized. - * \param src The SHA-1 context to clone from. This must be initialized. - * - */ -void mbedtls_sha1_clone( mbedtls_sha1_context *dst, - const mbedtls_sha1_context *src ); - -/** - * \brief This function starts a SHA-1 checksum calculation. - * - * \warning SHA-1 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - * \param ctx The SHA-1 context to initialize. This must be initialized. - * - * \return \c 0 on success. - * \return A negative error code on failure. - * - */ -int mbedtls_sha1_starts( mbedtls_sha1_context *ctx ); - -/** - * \brief This function feeds an input buffer into an ongoing SHA-1 - * checksum calculation. - * - * \warning SHA-1 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - * \param ctx The SHA-1 context. This must be initialized - * and have a hash operation started. - * \param input The buffer holding the input data. - * This must be a readable buffer of length \p ilen Bytes. - * \param ilen The length of the input data \p input in Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_sha1_update( mbedtls_sha1_context *ctx, - const unsigned char *input, - size_t ilen ); - -/** - * \brief This function finishes the SHA-1 operation, and writes - * the result to the output buffer. - * - * \warning SHA-1 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - * \param ctx The SHA-1 context to use. This must be initialized and - * have a hash operation started. - * \param output The SHA-1 checksum result. This must be a writable - * buffer of length \c 20 Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_sha1_finish( mbedtls_sha1_context *ctx, - unsigned char output[20] ); - -/** - * \brief SHA-1 process data block (internal use only). - * - * \warning SHA-1 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - * \param ctx The SHA-1 context to use. This must be initialized. - * \param data The data block being processed. This must be a - * readable buffer of length \c 64 Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - * - */ -int mbedtls_internal_sha1_process( mbedtls_sha1_context *ctx, - const unsigned char data[64] ); - -/** - * \brief This function calculates the SHA-1 checksum of a buffer. - * - * The function allocates the context, performs the - * calculation, and frees the context. - * - * The SHA-1 result is calculated as - * output = SHA-1(input buffer). - * - * \warning SHA-1 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - * \param input The buffer holding the input data. - * This must be a readable buffer of length \p ilen Bytes. - * \param ilen The length of the input data \p input in Bytes. - * \param output The SHA-1 checksum result. - * This must be a writable buffer of length \c 20 Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - * - */ -int mbedtls_sha1( const unsigned char *input, - size_t ilen, - unsigned char output[20] ); - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief The SHA-1 checkup routine. - * - * \warning SHA-1 is considered a weak message digest and its use - * constitutes a security risk. We recommend considering - * stronger message digests instead. - * - * \return \c 0 on success. - * \return \c 1 on failure. - * - */ -int mbedtls_sha1_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* mbedtls_sha1.h */ diff --git a/windows/mbedtls/sha256.h b/windows/mbedtls/sha256.h deleted file mode 100644 index 7f42c2c9..00000000 --- a/windows/mbedtls/sha256.h +++ /dev/null @@ -1,191 +0,0 @@ -/** - * \file sha256.h - * - * \brief This file contains SHA-224 and SHA-256 definitions and functions. - * - * The Secure Hash Algorithms 224 and 256 (SHA-224 and SHA-256) cryptographic - * hash functions are defined in FIPS 180-4: Secure Hash Standard (SHS). - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_SHA256_H -#define MBEDTLS_SHA256_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -#define MBEDTLS_ERR_SHA256_BAD_INPUT_DATA -0x0074 /**< SHA-256 input data was malformed. */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_SHA256_ALT) -// Regular implementation -// - -/** - * \brief The SHA-256 context structure. - * - * The structure is used both for SHA-256 and for SHA-224 - * checksum calculations. The choice between these two is - * made in the call to mbedtls_sha256_starts(). - */ -typedef struct mbedtls_sha256_context -{ - uint32_t MBEDTLS_PRIVATE(total)[2]; /*!< The number of Bytes processed. */ - uint32_t MBEDTLS_PRIVATE(state)[8]; /*!< The intermediate digest state. */ - unsigned char MBEDTLS_PRIVATE(buffer)[64]; /*!< The data block being processed. */ - int MBEDTLS_PRIVATE(is224); /*!< Determines which function to use: - 0: Use SHA-256, or 1: Use SHA-224. */ -} -mbedtls_sha256_context; - -#else /* MBEDTLS_SHA256_ALT */ -#include "sha256_alt.h" -#endif /* MBEDTLS_SHA256_ALT */ - -/** - * \brief This function initializes a SHA-256 context. - * - * \param ctx The SHA-256 context to initialize. This must not be \c NULL. - */ -void mbedtls_sha256_init( mbedtls_sha256_context *ctx ); - -/** - * \brief This function clears a SHA-256 context. - * - * \param ctx The SHA-256 context to clear. This may be \c NULL, in which - * case this function returns immediately. If it is not \c NULL, - * it must point to an initialized SHA-256 context. - */ -void mbedtls_sha256_free( mbedtls_sha256_context *ctx ); - -/** - * \brief This function clones the state of a SHA-256 context. - * - * \param dst The destination context. This must be initialized. - * \param src The context to clone. This must be initialized. - */ -void mbedtls_sha256_clone( mbedtls_sha256_context *dst, - const mbedtls_sha256_context *src ); - -/** - * \brief This function starts a SHA-224 or SHA-256 checksum - * calculation. - * - * \param ctx The context to use. This must be initialized. - * \param is224 This determines which function to use. This must be - * either \c 0 for SHA-256, or \c 1 for SHA-224. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_sha256_starts( mbedtls_sha256_context *ctx, int is224 ); - -/** - * \brief This function feeds an input buffer into an ongoing - * SHA-256 checksum calculation. - * - * \param ctx The SHA-256 context. This must be initialized - * and have a hash operation started. - * \param input The buffer holding the data. This must be a readable - * buffer of length \p ilen Bytes. - * \param ilen The length of the input data in Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_sha256_update( mbedtls_sha256_context *ctx, - const unsigned char *input, - size_t ilen ); - -/** - * \brief This function finishes the SHA-256 operation, and writes - * the result to the output buffer. - * - * \param ctx The SHA-256 context. This must be initialized - * and have a hash operation started. - * \param output The SHA-224 or SHA-256 checksum result. - * This must be a writable buffer of length \c 32 bytes - * for SHA-256, \c 28 bytes for SHA-224. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_sha256_finish( mbedtls_sha256_context *ctx, - unsigned char *output ); - -/** - * \brief This function processes a single data block within - * the ongoing SHA-256 computation. This function is for - * internal use only. - * - * \param ctx The SHA-256 context. This must be initialized. - * \param data The buffer holding one block of data. This must - * be a readable buffer of length \c 64 Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_internal_sha256_process( mbedtls_sha256_context *ctx, - const unsigned char data[64] ); - -/** - * \brief This function calculates the SHA-224 or SHA-256 - * checksum of a buffer. - * - * The function allocates the context, performs the - * calculation, and frees the context. - * - * The SHA-256 result is calculated as - * output = SHA-256(input buffer). - * - * \param input The buffer holding the data. This must be a readable - * buffer of length \p ilen Bytes. - * \param ilen The length of the input data in Bytes. - * \param output The SHA-224 or SHA-256 checksum result. - * This must be a writable buffer of length \c 32 bytes - * for SHA-256, \c 28 bytes for SHA-224. - * \param is224 Determines which function to use. This must be - * either \c 0 for SHA-256, or \c 1 for SHA-224. - */ -int mbedtls_sha256( const unsigned char *input, - size_t ilen, - unsigned char *output, - int is224 ); - -#if defined(MBEDTLS_SELF_TEST) - -/** - * \brief The SHA-224 and SHA-256 checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_sha256_self_test( int verbose ); - -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* mbedtls_sha256.h */ diff --git a/windows/mbedtls/sha512.h b/windows/mbedtls/sha512.h deleted file mode 100644 index c43b9075..00000000 --- a/windows/mbedtls/sha512.h +++ /dev/null @@ -1,204 +0,0 @@ -/** - * \file sha512.h - * \brief This file contains SHA-384 and SHA-512 definitions and functions. - * - * The Secure Hash Algorithms 384 and 512 (SHA-384 and SHA-512) cryptographic - * hash functions are defined in FIPS 180-4: Secure Hash Standard (SHS). - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_SHA512_H -#define MBEDTLS_SHA512_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include -#include - -#define MBEDTLS_ERR_SHA512_BAD_INPUT_DATA -0x0075 /**< SHA-512 input data was malformed. */ - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_SHA512_ALT) -// Regular implementation -// - -/** - * \brief The SHA-512 context structure. - * - * The structure is used both for SHA-384 and for SHA-512 - * checksum calculations. The choice between these two is - * made in the call to mbedtls_sha512_starts(). - */ -typedef struct mbedtls_sha512_context -{ - uint64_t MBEDTLS_PRIVATE(total)[2]; /*!< The number of Bytes processed. */ - uint64_t MBEDTLS_PRIVATE(state)[8]; /*!< The intermediate digest state. */ - unsigned char MBEDTLS_PRIVATE(buffer)[128]; /*!< The data block being processed. */ -#if defined(MBEDTLS_SHA384_C) - int MBEDTLS_PRIVATE(is384); /*!< Determines which function to use: - 0: Use SHA-512, or 1: Use SHA-384. */ -#endif -} -mbedtls_sha512_context; - -#else /* MBEDTLS_SHA512_ALT */ -#include "sha512_alt.h" -#endif /* MBEDTLS_SHA512_ALT */ - -/** - * \brief This function initializes a SHA-512 context. - * - * \param ctx The SHA-512 context to initialize. This must - * not be \c NULL. - */ -void mbedtls_sha512_init( mbedtls_sha512_context *ctx ); - -/** - * \brief This function clears a SHA-512 context. - * - * \param ctx The SHA-512 context to clear. This may be \c NULL, - * in which case this function does nothing. If it - * is not \c NULL, it must point to an initialized - * SHA-512 context. - */ -void mbedtls_sha512_free( mbedtls_sha512_context *ctx ); - -/** - * \brief This function clones the state of a SHA-512 context. - * - * \param dst The destination context. This must be initialized. - * \param src The context to clone. This must be initialized. - */ -void mbedtls_sha512_clone( mbedtls_sha512_context *dst, - const mbedtls_sha512_context *src ); - -/** - * \brief This function starts a SHA-384 or SHA-512 checksum - * calculation. - * - * \param ctx The SHA-512 context to use. This must be initialized. - * \param is384 Determines which function to use. This must be - * either \c 0 for SHA-512, or \c 1 for SHA-384. - * - * \note When \c MBEDTLS_SHA384_C is not defined, - * \p is384 must be \c 0, or the function will return - * #MBEDTLS_ERR_SHA512_BAD_INPUT_DATA. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_sha512_starts( mbedtls_sha512_context *ctx, int is384 ); - -/** - * \brief This function feeds an input buffer into an ongoing - * SHA-512 checksum calculation. - * - * \param ctx The SHA-512 context. This must be initialized - * and have a hash operation started. - * \param input The buffer holding the input data. This must - * be a readable buffer of length \p ilen Bytes. - * \param ilen The length of the input data in Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_sha512_update( mbedtls_sha512_context *ctx, - const unsigned char *input, - size_t ilen ); - -/** - * \brief This function finishes the SHA-512 operation, and writes - * the result to the output buffer. - * - * \param ctx The SHA-512 context. This must be initialized - * and have a hash operation started. - * \param output The SHA-384 or SHA-512 checksum result. - * This must be a writable buffer of length \c 64 bytes - * for SHA-512, \c 48 bytes for SHA-384. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_sha512_finish( mbedtls_sha512_context *ctx, - unsigned char *output ); - -/** - * \brief This function processes a single data block within - * the ongoing SHA-512 computation. - * This function is for internal use only. - * - * \param ctx The SHA-512 context. This must be initialized. - * \param data The buffer holding one block of data. This - * must be a readable buffer of length \c 128 Bytes. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_internal_sha512_process( mbedtls_sha512_context *ctx, - const unsigned char data[128] ); - -/** - * \brief This function calculates the SHA-512 or SHA-384 - * checksum of a buffer. - * - * The function allocates the context, performs the - * calculation, and frees the context. - * - * The SHA-512 result is calculated as - * output = SHA-512(input buffer). - * - * \param input The buffer holding the input data. This must be - * a readable buffer of length \p ilen Bytes. - * \param ilen The length of the input data in Bytes. - * \param output The SHA-384 or SHA-512 checksum result. - * This must be a writable buffer of length \c 64 bytes - * for SHA-512, \c 48 bytes for SHA-384. - * \param is384 Determines which function to use. This must be either - * \c 0 for SHA-512, or \c 1 for SHA-384. - * - * \note When \c MBEDTLS_SHA384_C is not defined, \p is384 must - * be \c 0, or the function will return - * #MBEDTLS_ERR_SHA512_BAD_INPUT_DATA. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_sha512( const unsigned char *input, - size_t ilen, - unsigned char *output, - int is384 ); - -#if defined(MBEDTLS_SELF_TEST) - - /** - * \brief The SHA-384 or SHA-512 checkup routine. - * - * \return \c 0 on success. - * \return \c 1 on failure. - */ -int mbedtls_sha512_self_test( int verbose ); -#endif /* MBEDTLS_SELF_TEST */ - -#ifdef __cplusplus -} -#endif - -#endif /* mbedtls_sha512.h */ diff --git a/windows/mbedtls/ssl.h b/windows/mbedtls/ssl.h deleted file mode 100644 index 3a14a583..00000000 --- a/windows/mbedtls/ssl.h +++ /dev/null @@ -1,4152 +0,0 @@ -/** - * \file ssl.h - * - * \brief SSL/TLS functions. - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_SSL_H -#define MBEDTLS_SSL_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/bignum.h" -#include "mbedtls/ecp.h" - -#include "mbedtls/ssl_ciphersuites.h" - -#if defined(MBEDTLS_X509_CRT_PARSE_C) -#include "mbedtls/x509_crt.h" -#include "mbedtls/x509_crl.h" -#endif - -#if defined(MBEDTLS_DHM_C) -#include "mbedtls/dhm.h" -#endif - -/* Adding guard for MBEDTLS_ECDSA_C to ensure no compile errors due - * to guards also being in ssl_srv.c and ssl_cli.c. There is a gap - * in functionality that access to ecdh_ctx structure is needed for - * MBEDTLS_ECDSA_C which does not seem correct. - */ -#if defined(MBEDTLS_ECDH_C) || defined(MBEDTLS_ECDSA_C) -#include "mbedtls/ecdh.h" -#endif - -#if defined(MBEDTLS_HAVE_TIME) -#include "mbedtls/platform_time.h" -#endif - -#if defined(MBEDTLS_USE_PSA_CRYPTO) -#include "psa/crypto.h" -#endif /* MBEDTLS_USE_PSA_CRYPTO */ - -/* - * SSL Error codes - */ -#define MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS -0x7000 /**< A cryptographic operation is in progress. Try again later. */ -#define MBEDTLS_ERR_SSL_FEATURE_UNAVAILABLE -0x7080 /**< The requested feature is not available. */ -#define MBEDTLS_ERR_SSL_BAD_INPUT_DATA -0x7100 /**< Bad input parameters to function. */ -#define MBEDTLS_ERR_SSL_INVALID_MAC -0x7180 /**< Verification of the message MAC failed. */ -#define MBEDTLS_ERR_SSL_INVALID_RECORD -0x7200 /**< An invalid SSL record was received. */ -#define MBEDTLS_ERR_SSL_CONN_EOF -0x7280 /**< The connection indicated an EOF. */ -#define MBEDTLS_ERR_SSL_DECODE_ERROR -0x7300 /**< A message could not be parsed due to a syntactic error. */ -/* Error space gap */ -#define MBEDTLS_ERR_SSL_NO_RNG -0x7400 /**< No RNG was provided to the SSL module. */ -#define MBEDTLS_ERR_SSL_NO_CLIENT_CERTIFICATE -0x7480 /**< No client certification received from the client, but required by the authentication mode. */ -#define MBEDTLS_ERR_SSL_UNSUPPORTED_EXTENSION -0x7500 /**< Client received an extended server hello containing an unsupported extension */ -#define MBEDTLS_ERR_SSL_NO_APPLICATION_PROTOCOL -0x7580 /**< No ALPN protocols supported that the client advertises */ -#define MBEDTLS_ERR_SSL_PRIVATE_KEY_REQUIRED -0x7600 /**< The own private key or pre-shared key is not set, but needed. */ -#define MBEDTLS_ERR_SSL_CA_CHAIN_REQUIRED -0x7680 /**< No CA Chain is set, but required to operate. */ -#define MBEDTLS_ERR_SSL_UNEXPECTED_MESSAGE -0x7700 /**< An unexpected message was received from our peer. */ -#define MBEDTLS_ERR_SSL_FATAL_ALERT_MESSAGE -0x7780 /**< A fatal alert message was received from our peer. */ -#define MBEDTLS_ERR_SSL_UNRECOGNIZED_NAME -0x7800 /**< No server could be identified matching the client's SNI. */ -#define MBEDTLS_ERR_SSL_PEER_CLOSE_NOTIFY -0x7880 /**< The peer notified us that the connection is going to be closed. */ -/* Error space gap */ -/* Error space gap */ -#define MBEDTLS_ERR_SSL_BAD_CERTIFICATE -0x7A00 /**< Processing of the Certificate handshake message failed. */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -#define MBEDTLS_ERR_SSL_ALLOC_FAILED -0x7F00 /**< Memory allocation failed */ -#define MBEDTLS_ERR_SSL_HW_ACCEL_FAILED -0x7F80 /**< Hardware acceleration function returned with error */ -#define MBEDTLS_ERR_SSL_HW_ACCEL_FALLTHROUGH -0x6F80 /**< Hardware acceleration function skipped / left alone data */ -#define MBEDTLS_ERR_SSL_BAD_PROTOCOL_VERSION -0x6E80 /**< Handshake protocol not within min/max boundaries */ -#define MBEDTLS_ERR_SSL_HANDSHAKE_FAILURE -0x6E00 /**< The handshake negotiation failed. */ -#define MBEDTLS_ERR_SSL_SESSION_TICKET_EXPIRED -0x6D80 /**< Session ticket has expired. */ -#define MBEDTLS_ERR_SSL_PK_TYPE_MISMATCH -0x6D00 /**< Public key type mismatch (eg, asked for RSA key exchange and presented EC key) */ -#define MBEDTLS_ERR_SSL_UNKNOWN_IDENTITY -0x6C80 /**< Unknown identity received (eg, PSK identity) */ -#define MBEDTLS_ERR_SSL_INTERNAL_ERROR -0x6C00 /**< Internal error (eg, unexpected failure in lower-level module) */ -#define MBEDTLS_ERR_SSL_COUNTER_WRAPPING -0x6B80 /**< A counter would wrap (eg, too many messages exchanged). */ -#define MBEDTLS_ERR_SSL_WAITING_SERVER_HELLO_RENEGO -0x6B00 /**< Unexpected message at ServerHello in renegotiation. */ -#define MBEDTLS_ERR_SSL_HELLO_VERIFY_REQUIRED -0x6A80 /**< DTLS client must retry for hello verification */ -#define MBEDTLS_ERR_SSL_BUFFER_TOO_SMALL -0x6A00 /**< A buffer is too small to receive or write a message */ -/* Error space gap */ -#define MBEDTLS_ERR_SSL_WANT_READ -0x6900 /**< No data of requested type currently available on underlying transport. */ -#define MBEDTLS_ERR_SSL_WANT_WRITE -0x6880 /**< Connection requires a write call. */ -#define MBEDTLS_ERR_SSL_TIMEOUT -0x6800 /**< The operation timed out. */ -#define MBEDTLS_ERR_SSL_CLIENT_RECONNECT -0x6780 /**< The client initiated a reconnect from the same port. */ -#define MBEDTLS_ERR_SSL_UNEXPECTED_RECORD -0x6700 /**< Record header looks valid but is not expected. */ -#define MBEDTLS_ERR_SSL_NON_FATAL -0x6680 /**< The alert message received indicates a non-fatal error. */ -#define MBEDTLS_ERR_SSL_ILLEGAL_PARAMETER -0x6600 /**< A field in a message was incorrect or inconsistent with other fields. */ -#define MBEDTLS_ERR_SSL_CONTINUE_PROCESSING -0x6580 /**< Internal-only message signaling that further message-processing should be done */ -#define MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS -0x6500 /**< The asynchronous operation is not completed yet. */ -#define MBEDTLS_ERR_SSL_EARLY_MESSAGE -0x6480 /**< Internal-only message signaling that a message arrived early. */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -/* Error space gap */ -#define MBEDTLS_ERR_SSL_UNEXPECTED_CID -0x6000 /**< An encrypted DTLS-frame with an unexpected CID was received. */ -#define MBEDTLS_ERR_SSL_VERSION_MISMATCH -0x5F00 /**< An operation failed due to an unexpected version or configuration. */ -#define MBEDTLS_ERR_SSL_BAD_CONFIG -0x5E80 /**< Invalid value in SSL config */ - -/* - * Various constants - */ - -/* These are the high an low bytes of ProtocolVersion as defined by: - * - RFC 5246: ProtocolVersion version = { 3, 3 }; // TLS v1.2 - * - RFC 8446: see section 4.2.1 - */ -#define MBEDTLS_SSL_MAJOR_VERSION_3 3 -#define MBEDTLS_SSL_MINOR_VERSION_3 3 /*!< TLS v1.2 */ -#define MBEDTLS_SSL_MINOR_VERSION_4 4 /*!< TLS v1.3 (experimental) */ - -#define MBEDTLS_SSL_TRANSPORT_STREAM 0 /*!< TLS */ -#define MBEDTLS_SSL_TRANSPORT_DATAGRAM 1 /*!< DTLS */ - -#define MBEDTLS_SSL_MAX_HOST_NAME_LEN 255 /*!< Maximum host name defined in RFC 1035 */ -#define MBEDTLS_SSL_MAX_ALPN_NAME_LEN 255 /*!< Maximum size in bytes of a protocol name in alpn ext., RFC 7301 */ - -#define MBEDTLS_SSL_MAX_ALPN_LIST_LEN 65535 /*!< Maximum size in bytes of list in alpn ext., RFC 7301 */ - -/* RFC 6066 section 4, see also mfl_code_to_length in ssl_tls.c - * NONE must be zero so that memset()ing structure to zero works */ -#define MBEDTLS_SSL_MAX_FRAG_LEN_NONE 0 /*!< don't use this extension */ -#define MBEDTLS_SSL_MAX_FRAG_LEN_512 1 /*!< MaxFragmentLength 2^9 */ -#define MBEDTLS_SSL_MAX_FRAG_LEN_1024 2 /*!< MaxFragmentLength 2^10 */ -#define MBEDTLS_SSL_MAX_FRAG_LEN_2048 3 /*!< MaxFragmentLength 2^11 */ -#define MBEDTLS_SSL_MAX_FRAG_LEN_4096 4 /*!< MaxFragmentLength 2^12 */ -#define MBEDTLS_SSL_MAX_FRAG_LEN_INVALID 5 /*!< first invalid value */ - -#define MBEDTLS_SSL_IS_CLIENT 0 -#define MBEDTLS_SSL_IS_SERVER 1 - -#define MBEDTLS_SSL_EXTENDED_MS_DISABLED 0 -#define MBEDTLS_SSL_EXTENDED_MS_ENABLED 1 - -#define MBEDTLS_SSL_CID_DISABLED 0 -#define MBEDTLS_SSL_CID_ENABLED 1 - -#define MBEDTLS_SSL_ETM_DISABLED 0 -#define MBEDTLS_SSL_ETM_ENABLED 1 - -#define MBEDTLS_SSL_COMPRESS_NULL 0 - -#define MBEDTLS_SSL_VERIFY_NONE 0 -#define MBEDTLS_SSL_VERIFY_OPTIONAL 1 -#define MBEDTLS_SSL_VERIFY_REQUIRED 2 -#define MBEDTLS_SSL_VERIFY_UNSET 3 /* Used only for sni_authmode */ - -#define MBEDTLS_SSL_LEGACY_RENEGOTIATION 0 -#define MBEDTLS_SSL_SECURE_RENEGOTIATION 1 - -#define MBEDTLS_SSL_RENEGOTIATION_DISABLED 0 -#define MBEDTLS_SSL_RENEGOTIATION_ENABLED 1 - -#define MBEDTLS_SSL_ANTI_REPLAY_DISABLED 0 -#define MBEDTLS_SSL_ANTI_REPLAY_ENABLED 1 - -#define MBEDTLS_SSL_RENEGOTIATION_NOT_ENFORCED -1 -#define MBEDTLS_SSL_RENEGO_MAX_RECORDS_DEFAULT 16 - -#define MBEDTLS_SSL_LEGACY_NO_RENEGOTIATION 0 -#define MBEDTLS_SSL_LEGACY_ALLOW_RENEGOTIATION 1 -#define MBEDTLS_SSL_LEGACY_BREAK_HANDSHAKE 2 - -#define MBEDTLS_SSL_TRUNC_HMAC_DISABLED 0 -#define MBEDTLS_SSL_TRUNC_HMAC_ENABLED 1 -#define MBEDTLS_SSL_TRUNCATED_HMAC_LEN 10 /* 80 bits, rfc 6066 section 7 */ - -#define MBEDTLS_SSL_SESSION_TICKETS_DISABLED 0 -#define MBEDTLS_SSL_SESSION_TICKETS_ENABLED 1 - -#define MBEDTLS_SSL_PRESET_DEFAULT 0 -#define MBEDTLS_SSL_PRESET_SUITEB 2 - -#define MBEDTLS_SSL_CERT_REQ_CA_LIST_ENABLED 1 -#define MBEDTLS_SSL_CERT_REQ_CA_LIST_DISABLED 0 - -#define MBEDTLS_SSL_DTLS_SRTP_MKI_UNSUPPORTED 0 -#define MBEDTLS_SSL_DTLS_SRTP_MKI_SUPPORTED 1 - -#define MBEDTLS_SSL_SRV_CIPHERSUITE_ORDER_CLIENT 1 -#define MBEDTLS_SSL_SRV_CIPHERSUITE_ORDER_SERVER 0 - -/* - * Default range for DTLS retransmission timer value, in milliseconds. - * RFC 6347 4.2.4.1 says from 1 second to 60 seconds. - */ -#define MBEDTLS_SSL_DTLS_TIMEOUT_DFL_MIN 1000 -#define MBEDTLS_SSL_DTLS_TIMEOUT_DFL_MAX 60000 - -/** - * \name SECTION: Module settings - * - * The configuration options you can set for this module are in this section. - * Either change them in mbedtls_config.h or define them on the compiler command line. - * \{ - */ - -/* - * Maximum fragment length in bytes, - * determines the size of each of the two internal I/O buffers. - * - * Note: the RFC defines the default size of SSL / TLS messages. If you - * change the value here, other clients / servers may not be able to - * communicate with you anymore. Only change this value if you control - * both sides of the connection and have it reduced at both sides, or - * if you're using the Max Fragment Length extension and you know all your - * peers are using it too! - */ -#if !defined(MBEDTLS_SSL_IN_CONTENT_LEN) -#define MBEDTLS_SSL_IN_CONTENT_LEN 16384 -#endif - -#if !defined(MBEDTLS_SSL_OUT_CONTENT_LEN) -#define MBEDTLS_SSL_OUT_CONTENT_LEN 16384 -#endif - -/* - * Maximum number of heap-allocated bytes for the purpose of - * DTLS handshake message reassembly and future message buffering. - */ -#if !defined(MBEDTLS_SSL_DTLS_MAX_BUFFERING) -#define MBEDTLS_SSL_DTLS_MAX_BUFFERING 32768 -#endif - -/* - * Maximum length of CIDs for incoming and outgoing messages. - */ -#if !defined(MBEDTLS_SSL_CID_IN_LEN_MAX) -#define MBEDTLS_SSL_CID_IN_LEN_MAX 32 -#endif - -#if !defined(MBEDTLS_SSL_CID_OUT_LEN_MAX) -#define MBEDTLS_SSL_CID_OUT_LEN_MAX 32 -#endif - -#if !defined(MBEDTLS_SSL_CID_TLS1_3_PADDING_GRANULARITY) -#define MBEDTLS_SSL_CID_TLS1_3_PADDING_GRANULARITY 16 -#endif - -/* \} name SECTION: Module settings */ - -/* - * Length of the verify data for secure renegotiation - */ -#define MBEDTLS_SSL_VERIFY_DATA_MAX_LEN 12 - -/* - * Signaling ciphersuite values (SCSV) - */ -#define MBEDTLS_SSL_EMPTY_RENEGOTIATION_INFO 0xFF /**< renegotiation info ext */ - -/* - * Supported Signature and Hash algorithms (For TLS 1.2) - * RFC 5246 section 7.4.1.4.1 - */ -#define MBEDTLS_SSL_HASH_NONE 0 -#define MBEDTLS_SSL_HASH_MD5 1 -#define MBEDTLS_SSL_HASH_SHA1 2 -#define MBEDTLS_SSL_HASH_SHA224 3 -#define MBEDTLS_SSL_HASH_SHA256 4 -#define MBEDTLS_SSL_HASH_SHA384 5 -#define MBEDTLS_SSL_HASH_SHA512 6 - -#define MBEDTLS_SSL_SIG_ANON 0 -#define MBEDTLS_SSL_SIG_RSA 1 -#define MBEDTLS_SSL_SIG_ECDSA 3 - -/* - * Client Certificate Types - * RFC 5246 section 7.4.4 plus RFC 4492 section 5.5 - */ -#define MBEDTLS_SSL_CERT_TYPE_RSA_SIGN 1 -#define MBEDTLS_SSL_CERT_TYPE_ECDSA_SIGN 64 - -/* - * Message, alert and handshake types - */ -#define MBEDTLS_SSL_MSG_CHANGE_CIPHER_SPEC 20 -#define MBEDTLS_SSL_MSG_ALERT 21 -#define MBEDTLS_SSL_MSG_HANDSHAKE 22 -#define MBEDTLS_SSL_MSG_APPLICATION_DATA 23 -#define MBEDTLS_SSL_MSG_CID 25 - -#define MBEDTLS_SSL_ALERT_LEVEL_WARNING 1 -#define MBEDTLS_SSL_ALERT_LEVEL_FATAL 2 - -#define MBEDTLS_SSL_ALERT_MSG_CLOSE_NOTIFY 0 /* 0x00 */ -#define MBEDTLS_SSL_ALERT_MSG_UNEXPECTED_MESSAGE 10 /* 0x0A */ -#define MBEDTLS_SSL_ALERT_MSG_BAD_RECORD_MAC 20 /* 0x14 */ -#define MBEDTLS_SSL_ALERT_MSG_DECRYPTION_FAILED 21 /* 0x15 */ -#define MBEDTLS_SSL_ALERT_MSG_RECORD_OVERFLOW 22 /* 0x16 */ -#define MBEDTLS_SSL_ALERT_MSG_DECOMPRESSION_FAILURE 30 /* 0x1E */ -#define MBEDTLS_SSL_ALERT_MSG_HANDSHAKE_FAILURE 40 /* 0x28 */ -#define MBEDTLS_SSL_ALERT_MSG_NO_CERT 41 /* 0x29 */ -#define MBEDTLS_SSL_ALERT_MSG_BAD_CERT 42 /* 0x2A */ -#define MBEDTLS_SSL_ALERT_MSG_UNSUPPORTED_CERT 43 /* 0x2B */ -#define MBEDTLS_SSL_ALERT_MSG_CERT_REVOKED 44 /* 0x2C */ -#define MBEDTLS_SSL_ALERT_MSG_CERT_EXPIRED 45 /* 0x2D */ -#define MBEDTLS_SSL_ALERT_MSG_CERT_UNKNOWN 46 /* 0x2E */ -#define MBEDTLS_SSL_ALERT_MSG_ILLEGAL_PARAMETER 47 /* 0x2F */ -#define MBEDTLS_SSL_ALERT_MSG_UNKNOWN_CA 48 /* 0x30 */ -#define MBEDTLS_SSL_ALERT_MSG_ACCESS_DENIED 49 /* 0x31 */ -#define MBEDTLS_SSL_ALERT_MSG_DECODE_ERROR 50 /* 0x32 */ -#define MBEDTLS_SSL_ALERT_MSG_DECRYPT_ERROR 51 /* 0x33 */ -#define MBEDTLS_SSL_ALERT_MSG_EXPORT_RESTRICTION 60 /* 0x3C */ -#define MBEDTLS_SSL_ALERT_MSG_PROTOCOL_VERSION 70 /* 0x46 */ -#define MBEDTLS_SSL_ALERT_MSG_INSUFFICIENT_SECURITY 71 /* 0x47 */ -#define MBEDTLS_SSL_ALERT_MSG_INTERNAL_ERROR 80 /* 0x50 */ -#define MBEDTLS_SSL_ALERT_MSG_INAPROPRIATE_FALLBACK 86 /* 0x56 */ -#define MBEDTLS_SSL_ALERT_MSG_USER_CANCELED 90 /* 0x5A */ -#define MBEDTLS_SSL_ALERT_MSG_NO_RENEGOTIATION 100 /* 0x64 */ -#define MBEDTLS_SSL_ALERT_MSG_UNSUPPORTED_EXT 110 /* 0x6E */ -#define MBEDTLS_SSL_ALERT_MSG_UNRECOGNIZED_NAME 112 /* 0x70 */ -#define MBEDTLS_SSL_ALERT_MSG_UNKNOWN_PSK_IDENTITY 115 /* 0x73 */ -#define MBEDTLS_SSL_ALERT_MSG_NO_APPLICATION_PROTOCOL 120 /* 0x78 */ - -#define MBEDTLS_SSL_HS_HELLO_REQUEST 0 -#define MBEDTLS_SSL_HS_CLIENT_HELLO 1 -#define MBEDTLS_SSL_HS_SERVER_HELLO 2 -#define MBEDTLS_SSL_HS_HELLO_VERIFY_REQUEST 3 -#define MBEDTLS_SSL_HS_NEW_SESSION_TICKET 4 -#define MBEDTLS_SSL_HS_CERTIFICATE 11 -#define MBEDTLS_SSL_HS_SERVER_KEY_EXCHANGE 12 -#define MBEDTLS_SSL_HS_CERTIFICATE_REQUEST 13 -#define MBEDTLS_SSL_HS_SERVER_HELLO_DONE 14 -#define MBEDTLS_SSL_HS_CERTIFICATE_VERIFY 15 -#define MBEDTLS_SSL_HS_CLIENT_KEY_EXCHANGE 16 -#define MBEDTLS_SSL_HS_FINISHED 20 - -/* - * TLS extensions - */ -#define MBEDTLS_TLS_EXT_SERVERNAME 0 -#define MBEDTLS_TLS_EXT_SERVERNAME_HOSTNAME 0 - -#define MBEDTLS_TLS_EXT_MAX_FRAGMENT_LENGTH 1 - -#define MBEDTLS_TLS_EXT_TRUNCATED_HMAC 4 - -#define MBEDTLS_TLS_EXT_SUPPORTED_ELLIPTIC_CURVES 10 -#define MBEDTLS_TLS_EXT_SUPPORTED_POINT_FORMATS 11 - -#define MBEDTLS_TLS_EXT_SIG_ALG 13 - -#define MBEDTLS_TLS_EXT_USE_SRTP 14 - -#define MBEDTLS_TLS_EXT_ALPN 16 - -#define MBEDTLS_TLS_EXT_ENCRYPT_THEN_MAC 22 /* 0x16 */ -#define MBEDTLS_TLS_EXT_EXTENDED_MASTER_SECRET 0x0017 /* 23 */ - -#define MBEDTLS_TLS_EXT_SESSION_TICKET 35 - -/* The value of the CID extension is still TBD as of - * draft-ietf-tls-dtls-connection-id-05 - * (https://tools.ietf.org/html/draft-ietf-tls-dtls-connection-id-05) */ -#define MBEDTLS_TLS_EXT_CID 254 /* TBD */ - -#define MBEDTLS_TLS_EXT_ECJPAKE_KKPP 256 /* experimental */ - -#define MBEDTLS_TLS_EXT_RENEGOTIATION_INFO 0xFF01 - -/* - * Size defines - */ -#if !defined(MBEDTLS_PSK_MAX_LEN) -#define MBEDTLS_PSK_MAX_LEN 32 /* 256 bits */ -#endif - -/* Dummy type used only for its size */ -union mbedtls_ssl_premaster_secret -{ -#if defined(MBEDTLS_KEY_EXCHANGE_RSA_ENABLED) - unsigned char _pms_rsa[48]; /* RFC 5246 8.1.1 */ -#endif -#if defined(MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED) - unsigned char _pms_dhm[MBEDTLS_MPI_MAX_SIZE]; /* RFC 5246 8.1.2 */ -#endif -#if defined(MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED) - unsigned char _pms_ecdh[MBEDTLS_ECP_MAX_BYTES]; /* RFC 4492 5.10 */ -#endif -#if defined(MBEDTLS_KEY_EXCHANGE_PSK_ENABLED) - unsigned char _pms_psk[4 + 2 * MBEDTLS_PSK_MAX_LEN]; /* RFC 4279 2 */ -#endif -#if defined(MBEDTLS_KEY_EXCHANGE_DHE_PSK_ENABLED) - unsigned char _pms_dhe_psk[4 + MBEDTLS_MPI_MAX_SIZE - + MBEDTLS_PSK_MAX_LEN]; /* RFC 4279 3 */ -#endif -#if defined(MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED) - unsigned char _pms_rsa_psk[52 + MBEDTLS_PSK_MAX_LEN]; /* RFC 4279 4 */ -#endif -#if defined(MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED) - unsigned char _pms_ecdhe_psk[4 + MBEDTLS_ECP_MAX_BYTES - + MBEDTLS_PSK_MAX_LEN]; /* RFC 5489 2 */ -#endif -#if defined(MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED) - unsigned char _pms_ecjpake[32]; /* Thread spec: SHA-256 output */ -#endif -}; - -#define MBEDTLS_PREMASTER_SIZE sizeof( union mbedtls_ssl_premaster_secret ) - -#ifdef __cplusplus -extern "C" { -#endif - -/* - * SSL state machine - */ -typedef enum -{ - MBEDTLS_SSL_HELLO_REQUEST, - MBEDTLS_SSL_CLIENT_HELLO, - MBEDTLS_SSL_SERVER_HELLO, - MBEDTLS_SSL_SERVER_CERTIFICATE, - MBEDTLS_SSL_SERVER_KEY_EXCHANGE, - MBEDTLS_SSL_CERTIFICATE_REQUEST, - MBEDTLS_SSL_SERVER_HELLO_DONE, - MBEDTLS_SSL_CLIENT_CERTIFICATE, - MBEDTLS_SSL_CLIENT_KEY_EXCHANGE, - MBEDTLS_SSL_CERTIFICATE_VERIFY, - MBEDTLS_SSL_CLIENT_CHANGE_CIPHER_SPEC, - MBEDTLS_SSL_CLIENT_FINISHED, - MBEDTLS_SSL_SERVER_CHANGE_CIPHER_SPEC, - MBEDTLS_SSL_SERVER_FINISHED, - MBEDTLS_SSL_FLUSH_BUFFERS, - MBEDTLS_SSL_HANDSHAKE_WRAPUP, - MBEDTLS_SSL_HANDSHAKE_OVER, - MBEDTLS_SSL_SERVER_NEW_SESSION_TICKET, - MBEDTLS_SSL_SERVER_HELLO_VERIFY_REQUEST_SENT, -} -mbedtls_ssl_states; - -/** - * \brief Callback type: send data on the network. - * - * \note That callback may be either blocking or non-blocking. - * - * \param ctx Context for the send callback (typically a file descriptor) - * \param buf Buffer holding the data to send - * \param len Length of the data to send - * - * \return The callback must return the number of bytes sent if any, - * or a non-zero error code. - * If performing non-blocking I/O, \c MBEDTLS_ERR_SSL_WANT_WRITE - * must be returned when the operation would block. - * - * \note The callback is allowed to send fewer bytes than requested. - * It must always return the number of bytes actually sent. - */ -typedef int mbedtls_ssl_send_t( void *ctx, - const unsigned char *buf, - size_t len ); - -/** - * \brief Callback type: receive data from the network. - * - * \note That callback may be either blocking or non-blocking. - * - * \param ctx Context for the receive callback (typically a file - * descriptor) - * \param buf Buffer to write the received data to - * \param len Length of the receive buffer - * - * \return The callback must return the number of bytes received, - * or a non-zero error code. - * If performing non-blocking I/O, \c MBEDTLS_ERR_SSL_WANT_READ - * must be returned when the operation would block. - * - * \note The callback may receive fewer bytes than the length of the - * buffer. It must always return the number of bytes actually - * received and written to the buffer. - */ -typedef int mbedtls_ssl_recv_t( void *ctx, - unsigned char *buf, - size_t len ); - -/** - * \brief Callback type: receive data from the network, with timeout - * - * \note That callback must block until data is received, or the - * timeout delay expires, or the operation is interrupted by a - * signal. - * - * \param ctx Context for the receive callback (typically a file descriptor) - * \param buf Buffer to write the received data to - * \param len Length of the receive buffer - * \param timeout Maximum nomber of millisecondes to wait for data - * 0 means no timeout (potentially waiting forever) - * - * \return The callback must return the number of bytes received, - * or a non-zero error code: - * \c MBEDTLS_ERR_SSL_TIMEOUT if the operation timed out, - * \c MBEDTLS_ERR_SSL_WANT_READ if interrupted by a signal. - * - * \note The callback may receive fewer bytes than the length of the - * buffer. It must always return the number of bytes actually - * received and written to the buffer. - */ -typedef int mbedtls_ssl_recv_timeout_t( void *ctx, - unsigned char *buf, - size_t len, - uint32_t timeout ); -/** - * \brief Callback type: set a pair of timers/delays to watch - * - * \param ctx Context pointer - * \param int_ms Intermediate delay in milliseconds - * \param fin_ms Final delay in milliseconds - * 0 cancels the current timer. - * - * \note This callback must at least store the necessary information - * for the associated \c mbedtls_ssl_get_timer_t callback to - * return correct information. - * - * \note If using a event-driven style of programming, an event must - * be generated when the final delay is passed. The event must - * cause a call to \c mbedtls_ssl_handshake() with the proper - * SSL context to be scheduled. Care must be taken to ensure - * that at most one such call happens at a time. - * - * \note Only one timer at a time must be running. Calling this - * function while a timer is running must cancel it. Cancelled - * timers must not generate any event. - */ -typedef void mbedtls_ssl_set_timer_t( void * ctx, - uint32_t int_ms, - uint32_t fin_ms ); - -/** - * \brief Callback type: get status of timers/delays - * - * \param ctx Context pointer - * - * \return This callback must return: - * -1 if cancelled (fin_ms == 0), - * 0 if none of the delays have passed, - * 1 if only the intermediate delay has passed, - * 2 if the final delay has passed. - */ -typedef int mbedtls_ssl_get_timer_t( void * ctx ); - -/* Defined below */ -typedef struct mbedtls_ssl_session mbedtls_ssl_session; -typedef struct mbedtls_ssl_context mbedtls_ssl_context; -typedef struct mbedtls_ssl_config mbedtls_ssl_config; - -/* Defined in library/ssl_misc.h */ -typedef struct mbedtls_ssl_transform mbedtls_ssl_transform; -typedef struct mbedtls_ssl_handshake_params mbedtls_ssl_handshake_params; -typedef struct mbedtls_ssl_sig_hash_set_t mbedtls_ssl_sig_hash_set_t; -#if defined(MBEDTLS_X509_CRT_PARSE_C) -typedef struct mbedtls_ssl_key_cert mbedtls_ssl_key_cert; -#endif -#if defined(MBEDTLS_SSL_PROTO_DTLS) -typedef struct mbedtls_ssl_flight_item mbedtls_ssl_flight_item; -#endif - -/** - * \brief Callback type: server-side session cache getter - * - * The session cache is logically a key value store, with - * keys being session IDs and values being instances of - * mbedtls_ssl_session. - * - * This callback retrieves an entry in this key-value store. - * - * \param data The address of the session cache structure to query. - * \param session_id The buffer holding the session ID to query. - * \param session_id_len The length of \p session_id in Bytes. - * \param session The address of the session structure to populate. - * It is initialized with mbdtls_ssl_session_init(), - * and the callback must always leave it in a state - * where it can safely be freed via - * mbedtls_ssl_session_free() independent of the - * return code of this function. - * - * \return \c 0 on success - * \return A non-zero return value on failure. - * - */ -typedef int mbedtls_ssl_cache_get_t( void *data, - unsigned char const *session_id, - size_t session_id_len, - mbedtls_ssl_session *session ); -/** - * \brief Callback type: server-side session cache setter - * - * The session cache is logically a key value store, with - * keys being session IDs and values being instances of - * mbedtls_ssl_session. - * - * This callback sets an entry in this key-value store. - * - * \param data The address of the session cache structure to modify. - * \param session_id The buffer holding the session ID to query. - * \param session_id_len The length of \p session_id in Bytes. - * \param session The address of the session to be stored in the - * session cache. - * - * \return \c 0 on success - * \return A non-zero return value on failure. - */ -typedef int mbedtls_ssl_cache_set_t( void *data, - unsigned char const *session_id, - size_t session_id_len, - const mbedtls_ssl_session *session ); - -#if defined(MBEDTLS_SSL_ASYNC_PRIVATE) -#if defined(MBEDTLS_X509_CRT_PARSE_C) -/** - * \brief Callback type: start external signature operation. - * - * This callback is called during an SSL handshake to start - * a signature decryption operation using an - * external processor. The parameter \p cert contains - * the public key; it is up to the callback function to - * determine how to access the associated private key. - * - * This function typically sends or enqueues a request, and - * does not wait for the operation to complete. This allows - * the handshake step to be non-blocking. - * - * The parameters \p ssl and \p cert are guaranteed to remain - * valid throughout the handshake. On the other hand, this - * function must save the contents of \p hash if the value - * is needed for later processing, because the \p hash buffer - * is no longer valid after this function returns. - * - * This function may call mbedtls_ssl_set_async_operation_data() - * to store an operation context for later retrieval - * by the resume or cancel callback. - * - * \note For RSA signatures, this function must produce output - * that is consistent with PKCS#1 v1.5 in the same way as - * mbedtls_rsa_pkcs1_sign(). Before the private key operation, - * apply the padding steps described in RFC 8017, section 9.2 - * "EMSA-PKCS1-v1_5" as follows. - * - If \p md_alg is #MBEDTLS_MD_NONE, apply the PKCS#1 v1.5 - * encoding, treating \p hash as the DigestInfo to be - * padded. In other words, apply EMSA-PKCS1-v1_5 starting - * from step 3, with `T = hash` and `tLen = hash_len`. - * - If `md_alg != MBEDTLS_MD_NONE`, apply the PKCS#1 v1.5 - * encoding, treating \p hash as the hash to be encoded and - * padded. In other words, apply EMSA-PKCS1-v1_5 starting - * from step 2, with `digestAlgorithm` obtained by calling - * mbedtls_oid_get_oid_by_md() on \p md_alg. - * - * \note For ECDSA signatures, the output format is the DER encoding - * `Ecdsa-Sig-Value` defined in - * [RFC 4492 section 5.4](https://tools.ietf.org/html/rfc4492#section-5.4). - * - * \param ssl The SSL connection instance. It should not be - * modified other than via - * mbedtls_ssl_set_async_operation_data(). - * \param cert Certificate containing the public key. - * In simple cases, this is one of the pointers passed to - * mbedtls_ssl_conf_own_cert() when configuring the SSL - * connection. However, if other callbacks are used, this - * property may not hold. For example, if an SNI callback - * is registered with mbedtls_ssl_conf_sni(), then - * this callback determines what certificate is used. - * \param md_alg Hash algorithm. - * \param hash Buffer containing the hash. This buffer is - * no longer valid when the function returns. - * \param hash_len Size of the \c hash buffer in bytes. - * - * \return 0 if the operation was started successfully and the SSL - * stack should call the resume callback immediately. - * \return #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS if the operation - * was started successfully and the SSL stack should return - * immediately without calling the resume callback yet. - * \return #MBEDTLS_ERR_SSL_HW_ACCEL_FALLTHROUGH if the external - * processor does not support this key. The SSL stack will - * use the private key object instead. - * \return Any other error indicates a fatal failure and is - * propagated up the call chain. The callback should - * use \c MBEDTLS_ERR_PK_xxx error codes, and must not - * use \c MBEDTLS_ERR_SSL_xxx error codes except as - * directed in the documentation of this callback. - */ -typedef int mbedtls_ssl_async_sign_t( mbedtls_ssl_context *ssl, - mbedtls_x509_crt *cert, - mbedtls_md_type_t md_alg, - const unsigned char *hash, - size_t hash_len ); - -/** - * \brief Callback type: start external decryption operation. - * - * This callback is called during an SSL handshake to start - * an RSA decryption operation using an - * external processor. The parameter \p cert contains - * the public key; it is up to the callback function to - * determine how to access the associated private key. - * - * This function typically sends or enqueues a request, and - * does not wait for the operation to complete. This allows - * the handshake step to be non-blocking. - * - * The parameters \p ssl and \p cert are guaranteed to remain - * valid throughout the handshake. On the other hand, this - * function must save the contents of \p input if the value - * is needed for later processing, because the \p input buffer - * is no longer valid after this function returns. - * - * This function may call mbedtls_ssl_set_async_operation_data() - * to store an operation context for later retrieval - * by the resume or cancel callback. - * - * \warning RSA decryption as used in TLS is subject to a potential - * timing side channel attack first discovered by Bleichenbacher - * in 1998. This attack can be remotely exploitable - * in practice. To avoid this attack, you must ensure that - * if the callback performs an RSA decryption, the time it - * takes to execute and return the result does not depend - * on whether the RSA decryption succeeded or reported - * invalid padding. - * - * \param ssl The SSL connection instance. It should not be - * modified other than via - * mbedtls_ssl_set_async_operation_data(). - * \param cert Certificate containing the public key. - * In simple cases, this is one of the pointers passed to - * mbedtls_ssl_conf_own_cert() when configuring the SSL - * connection. However, if other callbacks are used, this - * property may not hold. For example, if an SNI callback - * is registered with mbedtls_ssl_conf_sni(), then - * this callback determines what certificate is used. - * \param input Buffer containing the input ciphertext. This buffer - * is no longer valid when the function returns. - * \param input_len Size of the \p input buffer in bytes. - * - * \return 0 if the operation was started successfully and the SSL - * stack should call the resume callback immediately. - * \return #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS if the operation - * was started successfully and the SSL stack should return - * immediately without calling the resume callback yet. - * \return #MBEDTLS_ERR_SSL_HW_ACCEL_FALLTHROUGH if the external - * processor does not support this key. The SSL stack will - * use the private key object instead. - * \return Any other error indicates a fatal failure and is - * propagated up the call chain. The callback should - * use \c MBEDTLS_ERR_PK_xxx error codes, and must not - * use \c MBEDTLS_ERR_SSL_xxx error codes except as - * directed in the documentation of this callback. - */ -typedef int mbedtls_ssl_async_decrypt_t( mbedtls_ssl_context *ssl, - mbedtls_x509_crt *cert, - const unsigned char *input, - size_t input_len ); -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - -/** - * \brief Callback type: resume external operation. - * - * This callback is called during an SSL handshake to resume - * an external operation started by the - * ::mbedtls_ssl_async_sign_t or - * ::mbedtls_ssl_async_decrypt_t callback. - * - * This function typically checks the status of a pending - * request or causes the request queue to make progress, and - * does not wait for the operation to complete. This allows - * the handshake step to be non-blocking. - * - * This function may call mbedtls_ssl_get_async_operation_data() - * to retrieve an operation context set by the start callback. - * It may call mbedtls_ssl_set_async_operation_data() to modify - * this context. - * - * Note that when this function returns a status other than - * #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS, it must free any - * resources associated with the operation. - * - * \param ssl The SSL connection instance. It should not be - * modified other than via - * mbedtls_ssl_set_async_operation_data(). - * \param output Buffer containing the output (signature or decrypted - * data) on success. - * \param output_len On success, number of bytes written to \p output. - * \param output_size Size of the \p output buffer in bytes. - * - * \return 0 if output of the operation is available in the - * \p output buffer. - * \return #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS if the operation - * is still in progress. Subsequent requests for progress - * on the SSL connection will call the resume callback - * again. - * \return Any other error means that the operation is aborted. - * The SSL handshake is aborted. The callback should - * use \c MBEDTLS_ERR_PK_xxx error codes, and must not - * use \c MBEDTLS_ERR_SSL_xxx error codes except as - * directed in the documentation of this callback. - */ -typedef int mbedtls_ssl_async_resume_t( mbedtls_ssl_context *ssl, - unsigned char *output, - size_t *output_len, - size_t output_size ); - -/** - * \brief Callback type: cancel external operation. - * - * This callback is called if an SSL connection is closed - * while an asynchronous operation is in progress. Note that - * this callback is not called if the - * ::mbedtls_ssl_async_resume_t callback has run and has - * returned a value other than - * #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS, since in that case - * the asynchronous operation has already completed. - * - * This function may call mbedtls_ssl_get_async_operation_data() - * to retrieve an operation context set by the start callback. - * - * \param ssl The SSL connection instance. It should not be - * modified. - */ -typedef void mbedtls_ssl_async_cancel_t( mbedtls_ssl_context *ssl ); -#endif /* MBEDTLS_SSL_ASYNC_PRIVATE */ - -#if defined(MBEDTLS_KEY_EXCHANGE_WITH_CERT_ENABLED) && \ - !defined(MBEDTLS_SSL_KEEP_PEER_CERTIFICATE) -#define MBEDTLS_SSL_PEER_CERT_DIGEST_MAX_LEN 48 -#if defined(MBEDTLS_SHA256_C) -#define MBEDTLS_SSL_PEER_CERT_DIGEST_DFL_TYPE MBEDTLS_MD_SHA256 -#define MBEDTLS_SSL_PEER_CERT_DIGEST_DFL_LEN 32 -#elif defined(MBEDTLS_SHA384_C) -#define MBEDTLS_SSL_PEER_CERT_DIGEST_DFL_TYPE MBEDTLS_MD_SHA384 -#define MBEDTLS_SSL_PEER_CERT_DIGEST_DFL_LEN 48 -#elif defined(MBEDTLS_SHA1_C) -#define MBEDTLS_SSL_PEER_CERT_DIGEST_DFL_TYPE MBEDTLS_MD_SHA1 -#define MBEDTLS_SSL_PEER_CERT_DIGEST_DFL_LEN 20 -#else -/* This is already checked in check_config.h, but be sure. */ -#error "Bad configuration - need SHA-1, SHA-256 or SHA-512 enabled to compute digest of peer CRT." -#endif -#endif /* MBEDTLS_KEY_EXCHANGE_WITH_CERT_ENABLED && - !MBEDTLS_SSL_KEEP_PEER_CERTIFICATE */ - -#if defined(MBEDTLS_SSL_DTLS_SRTP) - -#define MBEDTLS_TLS_SRTP_MAX_MKI_LENGTH 255 -#define MBEDTLS_TLS_SRTP_MAX_PROFILE_LIST_LENGTH 4 -/* - * For code readability use a typedef for DTLS-SRTP profiles - * - * Use_srtp extension protection profiles values as defined in - * http://www.iana.org/assignments/srtp-protection/srtp-protection.xhtml - * - * Reminder: if this list is expanded mbedtls_ssl_check_srtp_profile_value - * must be updated too. - */ -#define MBEDTLS_TLS_SRTP_AES128_CM_HMAC_SHA1_80 ( (uint16_t) 0x0001) -#define MBEDTLS_TLS_SRTP_AES128_CM_HMAC_SHA1_32 ( (uint16_t) 0x0002) -#define MBEDTLS_TLS_SRTP_NULL_HMAC_SHA1_80 ( (uint16_t) 0x0005) -#define MBEDTLS_TLS_SRTP_NULL_HMAC_SHA1_32 ( (uint16_t) 0x0006) -/* This one is not iana defined, but for code readability. */ -#define MBEDTLS_TLS_SRTP_UNSET ( (uint16_t) 0x0000) - -typedef uint16_t mbedtls_ssl_srtp_profile; - -typedef struct mbedtls_dtls_srtp_info_t -{ - /*! The SRTP profile that was negotiated. */ - mbedtls_ssl_srtp_profile MBEDTLS_PRIVATE(chosen_dtls_srtp_profile); - /*! The length of mki_value. */ - uint16_t MBEDTLS_PRIVATE(mki_len); - /*! The mki_value used, with max size of 256 bytes. */ - unsigned char MBEDTLS_PRIVATE(mki_value)[MBEDTLS_TLS_SRTP_MAX_MKI_LENGTH]; -} -mbedtls_dtls_srtp_info; - -#endif /* MBEDTLS_SSL_DTLS_SRTP */ - -/* - * This structure is used for storing current session data. - * - * Note: when changing this definition, we need to check and update: - * - in tests/suites/test_suite_ssl.function: - * ssl_populate_session() and ssl_serialize_session_save_load() - * - in library/ssl_tls.c: - * mbedtls_ssl_session_init() and mbedtls_ssl_session_free() - * mbedtls_ssl_session_save() and ssl_session_load() - * ssl_session_copy() - */ -struct mbedtls_ssl_session -{ -#if defined(MBEDTLS_HAVE_TIME) - mbedtls_time_t MBEDTLS_PRIVATE(start); /*!< starting time */ -#endif - int MBEDTLS_PRIVATE(ciphersuite); /*!< chosen ciphersuite */ - int MBEDTLS_PRIVATE(compression); /*!< chosen compression */ - size_t MBEDTLS_PRIVATE(id_len); /*!< session id length */ - unsigned char MBEDTLS_PRIVATE(id)[32]; /*!< session identifier */ - unsigned char MBEDTLS_PRIVATE(master)[48]; /*!< the master secret */ - - unsigned char exported; - -#if defined(MBEDTLS_X509_CRT_PARSE_C) -#if defined(MBEDTLS_SSL_KEEP_PEER_CERTIFICATE) - mbedtls_x509_crt *MBEDTLS_PRIVATE(peer_cert); /*!< peer X.509 cert chain */ -#else /* MBEDTLS_SSL_KEEP_PEER_CERTIFICATE */ - /*! The digest of the peer's end-CRT. This must be kept to detect CRT - * changes during renegotiation, mitigating the triple handshake attack. */ - unsigned char *MBEDTLS_PRIVATE(peer_cert_digest); - size_t MBEDTLS_PRIVATE(peer_cert_digest_len); - mbedtls_md_type_t MBEDTLS_PRIVATE(peer_cert_digest_type); -#endif /* !MBEDTLS_SSL_KEEP_PEER_CERTIFICATE */ -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - uint32_t MBEDTLS_PRIVATE(verify_result); /*!< verification result */ - -#if defined(MBEDTLS_SSL_SESSION_TICKETS) && defined(MBEDTLS_SSL_CLI_C) - unsigned char *MBEDTLS_PRIVATE(ticket); /*!< RFC 5077 session ticket */ - size_t MBEDTLS_PRIVATE(ticket_len); /*!< session ticket length */ - uint32_t MBEDTLS_PRIVATE(ticket_lifetime); /*!< ticket lifetime hint */ -#endif /* MBEDTLS_SSL_SESSION_TICKETS && MBEDTLS_SSL_CLI_C */ - -#if defined(MBEDTLS_SSL_MAX_FRAGMENT_LENGTH) - unsigned char MBEDTLS_PRIVATE(mfl_code); /*!< MaxFragmentLength negotiated by peer */ -#endif /* MBEDTLS_SSL_MAX_FRAGMENT_LENGTH */ - -#if defined(MBEDTLS_SSL_ENCRYPT_THEN_MAC) - int MBEDTLS_PRIVATE(encrypt_then_mac); /*!< flag for EtM activation */ -#endif -}; - -/* - * Identifiers for PRFs used in various versions of TLS. - */ -typedef enum -{ - MBEDTLS_SSL_TLS_PRF_NONE, - MBEDTLS_SSL_TLS_PRF_SHA384, - MBEDTLS_SSL_TLS_PRF_SHA256, - MBEDTLS_SSL_HKDF_EXPAND_SHA384, - MBEDTLS_SSL_HKDF_EXPAND_SHA256 -} -mbedtls_tls_prf_types; - -#if defined(MBEDTLS_SSL_EXPORT_KEYS) -typedef enum -{ - MBEDTLS_SSL_KEY_EXPORT_TLS12_MASTER_SECRET = 0, -#if defined(MBEDTLS_SSL_PROTO_TLS1_3_EXPERIMENTAL) - MBEDTLS_SSL_KEY_EXPORT_TLS13_CLIENT_EARLY_SECRET, - MBEDTLS_SSL_KEY_EXPORT_TLS13_EARLY_EXPORTER_SECRET, - MBEDTLS_SSL_KEY_EXPORT_TLS13_CLIENT_HANDSHAKE_TRAFFIC_SECRET, - MBEDTLS_SSL_KEY_EXPORT_TLS13_SERVER_HANDSHAKE_TRAFFIC_SECRET, - MBEDTLS_SSL_KEY_EXPORT_TLS13_CLIENT_APPLICATION_TRAFFIC_SECRET, - MBEDTLS_SSL_KEY_EXPORT_TLS13_SERVER_APPLICATION_TRAFFIC_SECRET, -#endif /* MBEDTLS_SSL_PROTO_TLS1_3_EXPERIMENTAL */ -} mbedtls_ssl_key_export_type; - -/** - * \brief Callback type: Export key alongside random values for - * session identification, and PRF for - * implementation of TLS key exporters. - * - * \param p_expkey Context for the callback. - * \param type The type of the key that is being exported. - * \param secret The address of the buffer holding the secret - * that's being exporterd. - * \param secret_len The length of \p secret in bytes. - * \param client_random The client random bytes. - * \param server_random The server random bytes. - * \param tls_prf_type The identifier for the PRF used in the handshake - * to which the key belongs. - */ -typedef void mbedtls_ssl_export_keys_t( void *p_expkey, - mbedtls_ssl_key_export_type type, - const unsigned char *secret, - size_t secret_len, - const unsigned char client_random[32], - const unsigned char server_random[32], - mbedtls_tls_prf_types tls_prf_type ); -#endif /* MBEDTLS_SSL_EXPORT_KEYS */ - -/** - * SSL/TLS configuration to be shared between mbedtls_ssl_context structures. - */ -struct mbedtls_ssl_config -{ - /* Group items by size (largest first) to minimize padding overhead */ - - /* - * Pointers - */ - - /** Allowed ciphersuites for (D)TLS 1.2 (0-terminated) */ - const int *MBEDTLS_PRIVATE(ciphersuite_list); - - /** Callback for printing debug output */ - void (*MBEDTLS_PRIVATE(f_dbg))(void *, int, const char *, int, const char *); - void *MBEDTLS_PRIVATE(p_dbg); /*!< context for the debug function */ - - /** Callback for getting (pseudo-)random numbers */ - int (*MBEDTLS_PRIVATE(f_rng))(void *, unsigned char *, size_t); - void *MBEDTLS_PRIVATE(p_rng); /*!< context for the RNG function */ - - /** Callback to retrieve a session from the cache */ - mbedtls_ssl_cache_get_t *MBEDTLS_PRIVATE(f_get_cache); - /** Callback to store a session into the cache */ - mbedtls_ssl_cache_set_t *MBEDTLS_PRIVATE(f_set_cache); - void *MBEDTLS_PRIVATE(p_cache); /*!< context for cache callbacks */ - -#if defined(MBEDTLS_SSL_SERVER_NAME_INDICATION) - /** Callback for setting cert according to SNI extension */ - int (*MBEDTLS_PRIVATE(f_sni))(void *, mbedtls_ssl_context *, const unsigned char *, size_t); - void *MBEDTLS_PRIVATE(p_sni); /*!< context for SNI callback */ -#endif - -#if defined(MBEDTLS_X509_CRT_PARSE_C) - /** Callback to customize X.509 certificate chain verification */ - int (*MBEDTLS_PRIVATE(f_vrfy))(void *, mbedtls_x509_crt *, int, uint32_t *); - void *MBEDTLS_PRIVATE(p_vrfy); /*!< context for X.509 verify calllback */ -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_SOME_PSK_ENABLED) - /** Callback to retrieve PSK key from identity */ - int (*MBEDTLS_PRIVATE(f_psk))(void *, mbedtls_ssl_context *, const unsigned char *, size_t); - void *MBEDTLS_PRIVATE(p_psk); /*!< context for PSK callback */ -#endif - -#if defined(MBEDTLS_SSL_DTLS_HELLO_VERIFY) && defined(MBEDTLS_SSL_SRV_C) - /** Callback to create & write a cookie for ClientHello veirifcation */ - int (*MBEDTLS_PRIVATE(f_cookie_write))( void *, unsigned char **, unsigned char *, - const unsigned char *, size_t ); - /** Callback to verify validity of a ClientHello cookie */ - int (*MBEDTLS_PRIVATE(f_cookie_check))( void *, const unsigned char *, size_t, - const unsigned char *, size_t ); - void *MBEDTLS_PRIVATE(p_cookie); /*!< context for the cookie callbacks */ -#endif - -#if defined(MBEDTLS_SSL_SESSION_TICKETS) && defined(MBEDTLS_SSL_SRV_C) - /** Callback to create & write a session ticket */ - int (*MBEDTLS_PRIVATE(f_ticket_write))( void *, const mbedtls_ssl_session *, - unsigned char *, const unsigned char *, size_t *, uint32_t * ); - /** Callback to parse a session ticket into a session structure */ - int (*MBEDTLS_PRIVATE(f_ticket_parse))( void *, mbedtls_ssl_session *, unsigned char *, size_t); - void *MBEDTLS_PRIVATE(p_ticket); /*!< context for the ticket callbacks */ -#endif /* MBEDTLS_SSL_SESSION_TICKETS && MBEDTLS_SSL_SRV_C */ - -#if defined(MBEDTLS_SSL_DTLS_CONNECTION_ID) - size_t MBEDTLS_PRIVATE(cid_len); /*!< The length of CIDs for incoming DTLS records. */ -#endif /* MBEDTLS_SSL_DTLS_CONNECTION_ID */ - -#if defined(MBEDTLS_X509_CRT_PARSE_C) - const mbedtls_x509_crt_profile *MBEDTLS_PRIVATE(cert_profile); /*!< verification profile */ - mbedtls_ssl_key_cert *MBEDTLS_PRIVATE(key_cert); /*!< own certificate/key pair(s) */ - mbedtls_x509_crt *MBEDTLS_PRIVATE(ca_chain); /*!< trusted CAs */ - mbedtls_x509_crl *MBEDTLS_PRIVATE(ca_crl); /*!< trusted CAs CRLs */ -#if defined(MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK) - mbedtls_x509_crt_ca_cb_t MBEDTLS_PRIVATE(f_ca_cb); - void *MBEDTLS_PRIVATE(p_ca_cb); -#endif /* MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK */ -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - -#if defined(MBEDTLS_SSL_ASYNC_PRIVATE) -#if defined(MBEDTLS_X509_CRT_PARSE_C) - mbedtls_ssl_async_sign_t *MBEDTLS_PRIVATE(f_async_sign_start); /*!< start asynchronous signature operation */ - mbedtls_ssl_async_decrypt_t *MBEDTLS_PRIVATE(f_async_decrypt_start); /*!< start asynchronous decryption operation */ -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - mbedtls_ssl_async_resume_t *MBEDTLS_PRIVATE(f_async_resume); /*!< resume asynchronous operation */ - mbedtls_ssl_async_cancel_t *MBEDTLS_PRIVATE(f_async_cancel); /*!< cancel asynchronous operation */ - void *MBEDTLS_PRIVATE(p_async_config_data); /*!< Configuration data set by mbedtls_ssl_conf_async_private_cb(). */ -#endif /* MBEDTLS_SSL_ASYNC_PRIVATE */ - -#if defined(MBEDTLS_KEY_EXCHANGE_WITH_CERT_ENABLED) - const int *MBEDTLS_PRIVATE(sig_hashes); /*!< allowed signature hashes */ -#endif - -#if defined(MBEDTLS_ECP_C) - const mbedtls_ecp_group_id *MBEDTLS_PRIVATE(curve_list); /*!< allowed curves */ -#endif - -#if defined(MBEDTLS_DHM_C) - mbedtls_mpi MBEDTLS_PRIVATE(dhm_P); /*!< prime modulus for DHM */ - mbedtls_mpi MBEDTLS_PRIVATE(dhm_G); /*!< generator for DHM */ -#endif - -#if defined(MBEDTLS_KEY_EXCHANGE_SOME_PSK_ENABLED) - -#if defined(MBEDTLS_USE_PSA_CRYPTO) - psa_key_id_t MBEDTLS_PRIVATE(psk_opaque); /*!< PSA key slot holding opaque PSK. This field - * should only be set via - * mbedtls_ssl_conf_psk_opaque(). - * If either no PSK or a raw PSK have been - * configured, this has value \c 0. - */ -#endif /* MBEDTLS_USE_PSA_CRYPTO */ - - unsigned char *MBEDTLS_PRIVATE(psk); /*!< The raw pre-shared key. This field should - * only be set via mbedtls_ssl_conf_psk(). - * If either no PSK or an opaque PSK - * have been configured, this has value NULL. */ - size_t MBEDTLS_PRIVATE(psk_len); /*!< The length of the raw pre-shared key. - * This field should only be set via - * mbedtls_ssl_conf_psk(). - * Its value is non-zero if and only if - * \c psk is not \c NULL. */ - - unsigned char *MBEDTLS_PRIVATE(psk_identity); /*!< The PSK identity for PSK negotiation. - * This field should only be set via - * mbedtls_ssl_conf_psk(). - * This is set if and only if either - * \c psk or \c psk_opaque are set. */ - size_t MBEDTLS_PRIVATE(psk_identity_len);/*!< The length of PSK identity. - * This field should only be set via - * mbedtls_ssl_conf_psk(). - * Its value is non-zero if and only if - * \c psk is not \c NULL or \c psk_opaque - * is not \c 0. */ -#endif /* MBEDTLS_KEY_EXCHANGE_SOME_PSK_ENABLED */ - -#if defined(MBEDTLS_SSL_ALPN) - const char **MBEDTLS_PRIVATE(alpn_list); /*!< ordered list of protocols */ -#endif - -#if defined(MBEDTLS_SSL_DTLS_SRTP) - /*! ordered list of supported srtp profile */ - const mbedtls_ssl_srtp_profile *MBEDTLS_PRIVATE(dtls_srtp_profile_list); - /*! number of supported profiles */ - size_t MBEDTLS_PRIVATE(dtls_srtp_profile_list_len); -#endif /* MBEDTLS_SSL_DTLS_SRTP */ - - /* - * Numerical settings (int then char) - */ - - uint32_t MBEDTLS_PRIVATE(read_timeout); /*!< timeout for mbedtls_ssl_read (ms) */ - -#if defined(MBEDTLS_SSL_PROTO_DTLS) - uint32_t MBEDTLS_PRIVATE(hs_timeout_min); /*!< initial value of the handshake - retransmission timeout (ms) */ - uint32_t MBEDTLS_PRIVATE(hs_timeout_max); /*!< maximum value of the handshake - retransmission timeout (ms) */ -#endif - -#if defined(MBEDTLS_SSL_RENEGOTIATION) - int MBEDTLS_PRIVATE(renego_max_records); /*!< grace period for renegotiation */ - unsigned char MBEDTLS_PRIVATE(renego_period)[8]; /*!< value of the record counters - that triggers renegotiation */ -#endif - - unsigned int MBEDTLS_PRIVATE(badmac_limit); /*!< limit of records with a bad MAC */ - -#if defined(MBEDTLS_DHM_C) && defined(MBEDTLS_SSL_CLI_C) - unsigned int MBEDTLS_PRIVATE(dhm_min_bitlen); /*!< min. bit length of the DHM prime */ -#endif - - unsigned char MBEDTLS_PRIVATE(max_major_ver); /*!< max. major version used */ - unsigned char MBEDTLS_PRIVATE(max_minor_ver); /*!< max. minor version used */ - unsigned char MBEDTLS_PRIVATE(min_major_ver); /*!< min. major version used */ - unsigned char MBEDTLS_PRIVATE(min_minor_ver); /*!< min. minor version used */ - - /* - * Flags (bitfields) - */ - - unsigned int MBEDTLS_PRIVATE(endpoint) : 1; /*!< 0: client, 1: server */ - unsigned int MBEDTLS_PRIVATE(transport) : 1; /*!< stream (TLS) or datagram (DTLS) */ - unsigned int MBEDTLS_PRIVATE(authmode) : 2; /*!< MBEDTLS_SSL_VERIFY_XXX */ - /* needed even with renego disabled for LEGACY_BREAK_HANDSHAKE */ - unsigned int MBEDTLS_PRIVATE(allow_legacy_renegotiation) : 2 ; /*!< MBEDTLS_LEGACY_XXX */ -#if defined(MBEDTLS_SSL_MAX_FRAGMENT_LENGTH) - unsigned int MBEDTLS_PRIVATE(mfl_code) : 3; /*!< desired fragment length */ -#endif -#if defined(MBEDTLS_SSL_ENCRYPT_THEN_MAC) - unsigned int MBEDTLS_PRIVATE(encrypt_then_mac) : 1 ; /*!< negotiate encrypt-then-mac? */ -#endif -#if defined(MBEDTLS_SSL_EXTENDED_MASTER_SECRET) - unsigned int MBEDTLS_PRIVATE(extended_ms) : 1; /*!< negotiate extended master secret? */ -#endif -#if defined(MBEDTLS_SSL_DTLS_ANTI_REPLAY) - unsigned int MBEDTLS_PRIVATE(anti_replay) : 1; /*!< detect and prevent replay? */ -#endif -#if defined(MBEDTLS_SSL_RENEGOTIATION) - unsigned int MBEDTLS_PRIVATE(disable_renegotiation) : 1; /*!< disable renegotiation? */ -#endif -#if defined(MBEDTLS_SSL_SESSION_TICKETS) - unsigned int MBEDTLS_PRIVATE(session_tickets) : 1; /*!< use session tickets? */ -#endif -#if defined(MBEDTLS_SSL_SRV_C) - unsigned int MBEDTLS_PRIVATE(cert_req_ca_list) : 1; /*!< enable sending CA list in - Certificate Request messages? */ - unsigned int respect_cli_pref : 1; /*!< pick the ciphersuite according to - the client's preferences rather - than ours */ -#endif -#if defined(MBEDTLS_SSL_DTLS_CONNECTION_ID) - unsigned int MBEDTLS_PRIVATE(ignore_unexpected_cid) : 1; /*!< Determines whether DTLS - * record with unexpected CID - * should lead to failure. */ -#endif /* MBEDTLS_SSL_DTLS_CONNECTION_ID */ -#if defined(MBEDTLS_SSL_DTLS_SRTP) - unsigned int MBEDTLS_PRIVATE(dtls_srtp_mki_support) : 1; /* support having mki_value - in the use_srtp extension */ -#endif -}; - -struct mbedtls_ssl_context -{ - const mbedtls_ssl_config *MBEDTLS_PRIVATE(conf); /*!< configuration information */ - - /* - * Miscellaneous - */ - int MBEDTLS_PRIVATE(state); /*!< SSL handshake: current state */ -#if defined(MBEDTLS_SSL_RENEGOTIATION) - int MBEDTLS_PRIVATE(renego_status); /*!< Initial, in progress, pending? */ - int MBEDTLS_PRIVATE(renego_records_seen); /*!< Records since renego request, or with DTLS, - number of retransmissions of request if - renego_max_records is < 0 */ -#endif /* MBEDTLS_SSL_RENEGOTIATION */ - - int MBEDTLS_PRIVATE(major_ver); /*!< equal to MBEDTLS_SSL_MAJOR_VERSION_3 */ - int MBEDTLS_PRIVATE(minor_ver); /*!< one of MBEDTLS_SSL_MINOR_VERSION_x macros */ - unsigned MBEDTLS_PRIVATE(badmac_seen); /*!< records with a bad MAC received */ - -#if defined(MBEDTLS_X509_CRT_PARSE_C) - /** Callback to customize X.509 certificate chain verification */ - int (*MBEDTLS_PRIVATE(f_vrfy))(void *, mbedtls_x509_crt *, int, uint32_t *); - void *MBEDTLS_PRIVATE(p_vrfy); /*!< context for X.509 verify callback */ -#endif - - mbedtls_ssl_send_t *MBEDTLS_PRIVATE(f_send); /*!< Callback for network send */ - mbedtls_ssl_recv_t *MBEDTLS_PRIVATE(f_recv); /*!< Callback for network receive */ - mbedtls_ssl_recv_timeout_t *MBEDTLS_PRIVATE(f_recv_timeout); - /*!< Callback for network receive with timeout */ - - void *MBEDTLS_PRIVATE(p_bio); /*!< context for I/O operations */ - - /* - * Session layer - */ - mbedtls_ssl_session *MBEDTLS_PRIVATE(session_in); /*!< current session data (in) */ - mbedtls_ssl_session *MBEDTLS_PRIVATE(session_out); /*!< current session data (out) */ - mbedtls_ssl_session *MBEDTLS_PRIVATE(session); /*!< negotiated session data */ - mbedtls_ssl_session *MBEDTLS_PRIVATE(session_negotiate); /*!< session data in negotiation */ - - mbedtls_ssl_handshake_params *MBEDTLS_PRIVATE(handshake); /*!< params required only during - the handshake process */ - - /* - * Record layer transformations - */ - mbedtls_ssl_transform *MBEDTLS_PRIVATE(transform_in); /*!< current transform params (in) */ - mbedtls_ssl_transform *MBEDTLS_PRIVATE(transform_out); /*!< current transform params (in) */ - mbedtls_ssl_transform *MBEDTLS_PRIVATE(transform); /*!< negotiated transform params */ - mbedtls_ssl_transform *MBEDTLS_PRIVATE(transform_negotiate); /*!< transform params in negotiation */ - - /* - * Timers - */ - void *MBEDTLS_PRIVATE(p_timer); /*!< context for the timer callbacks */ - - mbedtls_ssl_set_timer_t *MBEDTLS_PRIVATE(f_set_timer); /*!< set timer callback */ - mbedtls_ssl_get_timer_t *MBEDTLS_PRIVATE(f_get_timer); /*!< get timer callback */ - - /* - * Record layer (incoming data) - */ - unsigned char *MBEDTLS_PRIVATE(in_buf); /*!< input buffer */ - unsigned char *MBEDTLS_PRIVATE(in_ctr); /*!< 64-bit incoming message counter - TLS: maintained by us - DTLS: read from peer */ - unsigned char *MBEDTLS_PRIVATE(in_hdr); /*!< start of record header */ -#if defined(MBEDTLS_SSL_DTLS_CONNECTION_ID) - unsigned char *MBEDTLS_PRIVATE(in_cid); /*!< The start of the CID; - * (the end is marked by in_len). */ -#endif /* MBEDTLS_SSL_DTLS_CONNECTION_ID */ - unsigned char *MBEDTLS_PRIVATE(in_len); /*!< two-bytes message length field */ - unsigned char *MBEDTLS_PRIVATE(in_iv); /*!< ivlen-byte IV */ - unsigned char *MBEDTLS_PRIVATE(in_msg); /*!< message contents (in_iv+ivlen) */ - unsigned char *MBEDTLS_PRIVATE(in_offt); /*!< read offset in application data */ - - int MBEDTLS_PRIVATE(in_msgtype); /*!< record header: message type */ - size_t MBEDTLS_PRIVATE(in_msglen); /*!< record header: message length */ - size_t MBEDTLS_PRIVATE(in_left); /*!< amount of data read so far */ -#if defined(MBEDTLS_SSL_VARIABLE_BUFFER_LENGTH) - size_t MBEDTLS_PRIVATE(in_buf_len); /*!< length of input buffer */ -#endif -#if defined(MBEDTLS_SSL_PROTO_DTLS) - uint16_t MBEDTLS_PRIVATE(in_epoch); /*!< DTLS epoch for incoming records */ - size_t MBEDTLS_PRIVATE(next_record_offset); /*!< offset of the next record in datagram - (equal to in_left if none) */ -#endif /* MBEDTLS_SSL_PROTO_DTLS */ -#if defined(MBEDTLS_SSL_DTLS_ANTI_REPLAY) - uint64_t MBEDTLS_PRIVATE(in_window_top); /*!< last validated record seq_num */ - uint64_t MBEDTLS_PRIVATE(in_window); /*!< bitmask for replay detection */ -#endif /* MBEDTLS_SSL_DTLS_ANTI_REPLAY */ - - size_t MBEDTLS_PRIVATE(in_hslen); /*!< current handshake message length, - including the handshake header */ - int MBEDTLS_PRIVATE(nb_zero); /*!< # of 0-length encrypted messages */ - - int MBEDTLS_PRIVATE(keep_current_message); /*!< drop or reuse current message - on next call to record layer? */ - -#if defined(MBEDTLS_SSL_PROTO_DTLS) - uint8_t MBEDTLS_PRIVATE(disable_datagram_packing); /*!< Disable packing multiple records - * within a single datagram. */ -#endif /* MBEDTLS_SSL_PROTO_DTLS */ - - /* - * Record layer (outgoing data) - */ - unsigned char *MBEDTLS_PRIVATE(out_buf); /*!< output buffer */ - unsigned char *MBEDTLS_PRIVATE(out_ctr); /*!< 64-bit outgoing message counter */ - unsigned char *MBEDTLS_PRIVATE(out_hdr); /*!< start of record header */ -#if defined(MBEDTLS_SSL_DTLS_CONNECTION_ID) - unsigned char *MBEDTLS_PRIVATE(out_cid); /*!< The start of the CID; - * (the end is marked by in_len). */ -#endif /* MBEDTLS_SSL_DTLS_CONNECTION_ID */ - unsigned char *MBEDTLS_PRIVATE(out_len); /*!< two-bytes message length field */ - unsigned char *MBEDTLS_PRIVATE(out_iv); /*!< ivlen-byte IV */ - unsigned char *MBEDTLS_PRIVATE(out_msg); /*!< message contents (out_iv+ivlen) */ - - int MBEDTLS_PRIVATE(out_msgtype); /*!< record header: message type */ - size_t MBEDTLS_PRIVATE(out_msglen); /*!< record header: message length */ - size_t MBEDTLS_PRIVATE(out_left); /*!< amount of data not yet written */ -#if defined(MBEDTLS_SSL_VARIABLE_BUFFER_LENGTH) - size_t MBEDTLS_PRIVATE(out_buf_len); /*!< length of output buffer */ -#endif - - unsigned char MBEDTLS_PRIVATE(cur_out_ctr)[8]; /*!< Outgoing record sequence number. */ - -#if defined(MBEDTLS_SSL_PROTO_DTLS) - uint16_t MBEDTLS_PRIVATE(mtu); /*!< path mtu, used to fragment outgoing messages */ -#endif /* MBEDTLS_SSL_PROTO_DTLS */ - - /* - * PKI layer - */ - int MBEDTLS_PRIVATE(client_auth); /*!< flag for client auth. */ - - /* - * User settings - */ -#if defined(MBEDTLS_X509_CRT_PARSE_C) - char *MBEDTLS_PRIVATE(hostname); /*!< expected peer CN for verification - (and SNI if available) */ -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - -#if defined(MBEDTLS_SSL_ALPN) - const char *MBEDTLS_PRIVATE(alpn_chosen); /*!< negotiated protocol */ -#endif /* MBEDTLS_SSL_ALPN */ - -#if defined(MBEDTLS_SSL_DTLS_SRTP) - /* - * use_srtp extension - */ - mbedtls_dtls_srtp_info MBEDTLS_PRIVATE(dtls_srtp_info); -#endif /* MBEDTLS_SSL_DTLS_SRTP */ - - /* - * Information for DTLS hello verify - */ -#if defined(MBEDTLS_SSL_DTLS_HELLO_VERIFY) && defined(MBEDTLS_SSL_SRV_C) - unsigned char *MBEDTLS_PRIVATE(cli_id); /*!< transport-level ID of the client */ - size_t MBEDTLS_PRIVATE(cli_id_len); /*!< length of cli_id */ -#endif /* MBEDTLS_SSL_DTLS_HELLO_VERIFY && MBEDTLS_SSL_SRV_C */ - - /* - * Secure renegotiation - */ - /* needed to know when to send extension on server */ - int MBEDTLS_PRIVATE(secure_renegotiation); /*!< does peer support legacy or - secure renegotiation */ -#if defined(MBEDTLS_SSL_RENEGOTIATION) - size_t MBEDTLS_PRIVATE(verify_data_len); /*!< length of verify data stored */ - char MBEDTLS_PRIVATE(own_verify_data)[MBEDTLS_SSL_VERIFY_DATA_MAX_LEN]; /*!< previous handshake verify data */ - char MBEDTLS_PRIVATE(peer_verify_data)[MBEDTLS_SSL_VERIFY_DATA_MAX_LEN]; /*!< previous handshake verify data */ -#endif /* MBEDTLS_SSL_RENEGOTIATION */ - -#if defined(MBEDTLS_SSL_DTLS_CONNECTION_ID) - /* CID configuration to use in subsequent handshakes. */ - - /*! The next incoming CID, chosen by the user and applying to - * all subsequent handshakes. This may be different from the - * CID currently used in case the user has re-configured the CID - * after an initial handshake. */ - unsigned char MBEDTLS_PRIVATE(own_cid)[ MBEDTLS_SSL_CID_IN_LEN_MAX ]; - uint8_t MBEDTLS_PRIVATE(own_cid_len); /*!< The length of \c own_cid. */ - uint8_t MBEDTLS_PRIVATE(negotiate_cid); /*!< This indicates whether the CID extension should - * be negotiated in the next handshake or not. - * Possible values are #MBEDTLS_SSL_CID_ENABLED - * and #MBEDTLS_SSL_CID_DISABLED. */ -#endif /* MBEDTLS_SSL_DTLS_CONNECTION_ID */ - -#if defined(MBEDTLS_SSL_EXPORT_KEYS) - /** Callback to export key block and master secret */ - mbedtls_ssl_export_keys_t *MBEDTLS_PRIVATE(f_export_keys); - void *MBEDTLS_PRIVATE(p_export_keys); /*!< context for key export callback */ -#endif -}; - -/** - * \brief Return the name of the ciphersuite associated with the - * given ID - * - * \param ciphersuite_id SSL ciphersuite ID - * - * \return a string containing the ciphersuite name - */ -const char *mbedtls_ssl_get_ciphersuite_name( const int ciphersuite_id ); - -/** - * \brief Return the ID of the ciphersuite associated with the - * given name - * - * \param ciphersuite_name SSL ciphersuite name - * - * \return the ID with the ciphersuite or 0 if not found - */ -int mbedtls_ssl_get_ciphersuite_id( const char *ciphersuite_name ); - -/** - * \brief Initialize an SSL context - * Just makes the context ready for mbedtls_ssl_setup() or - * mbedtls_ssl_free() - * - * \param ssl SSL context - */ -void mbedtls_ssl_init( mbedtls_ssl_context *ssl ); - -/** - * \brief Set up an SSL context for use - * - * \note No copy of the configuration context is made, it can be - * shared by many mbedtls_ssl_context structures. - * - * \warning The conf structure will be accessed during the session. - * It must not be modified or freed as long as the session - * is active. - * - * \warning This function must be called exactly once per context. - * Calling mbedtls_ssl_setup again is not supported, even - * if no session is active. - * - * \param ssl SSL context - * \param conf SSL configuration to use - * - * \return 0 if successful, or MBEDTLS_ERR_SSL_ALLOC_FAILED if - * memory allocation failed - */ -int mbedtls_ssl_setup( mbedtls_ssl_context *ssl, - const mbedtls_ssl_config *conf ); - -/** - * \brief Reset an already initialized SSL context for re-use - * while retaining application-set variables, function - * pointers and data. - * - * \param ssl SSL context - * \return 0 if successful, or MBEDTLS_ERR_SSL_ALLOC_FAILED or - MBEDTLS_ERR_SSL_HW_ACCEL_FAILED - */ -int mbedtls_ssl_session_reset( mbedtls_ssl_context *ssl ); - -/** - * \brief Set the current endpoint type - * - * \param conf SSL configuration - * \param endpoint must be MBEDTLS_SSL_IS_CLIENT or MBEDTLS_SSL_IS_SERVER - */ -void mbedtls_ssl_conf_endpoint( mbedtls_ssl_config *conf, int endpoint ); - -/** - * \brief Set the transport type (TLS or DTLS). - * Default: TLS - * - * \note For DTLS, you must either provide a recv callback that - * doesn't block, or one that handles timeouts, see - * \c mbedtls_ssl_set_bio(). You also need to provide timer - * callbacks with \c mbedtls_ssl_set_timer_cb(). - * - * \param conf SSL configuration - * \param transport transport type: - * MBEDTLS_SSL_TRANSPORT_STREAM for TLS, - * MBEDTLS_SSL_TRANSPORT_DATAGRAM for DTLS. - */ -void mbedtls_ssl_conf_transport( mbedtls_ssl_config *conf, int transport ); - -/** - * \brief Set the certificate verification mode - * Default: NONE on server, REQUIRED on client - * - * \param conf SSL configuration - * \param authmode can be: - * - * MBEDTLS_SSL_VERIFY_NONE: peer certificate is not checked - * (default on server) - * (insecure on client) - * - * MBEDTLS_SSL_VERIFY_OPTIONAL: peer certificate is checked, however the - * handshake continues even if verification failed; - * mbedtls_ssl_get_verify_result() can be called after the - * handshake is complete. - * - * MBEDTLS_SSL_VERIFY_REQUIRED: peer *must* present a valid certificate, - * handshake is aborted if verification failed. - * (default on client) - * - * \note On client, MBEDTLS_SSL_VERIFY_REQUIRED is the recommended mode. - * With MBEDTLS_SSL_VERIFY_OPTIONAL, the user needs to call mbedtls_ssl_get_verify_result() at - * the right time(s), which may not be obvious, while REQUIRED always perform - * the verification as soon as possible. For example, REQUIRED was protecting - * against the "triple handshake" attack even before it was found. - */ -void mbedtls_ssl_conf_authmode( mbedtls_ssl_config *conf, int authmode ); - -#if defined(MBEDTLS_X509_CRT_PARSE_C) -/** - * \brief Set the verification callback (Optional). - * - * If set, the provided verify callback is called for each - * certificate in the peer's CRT chain, including the trusted - * root. For more information, please see the documentation of - * \c mbedtls_x509_crt_verify(). - * - * \note For per context callbacks and contexts, please use - * mbedtls_ssl_set_verify() instead. - * - * \param conf The SSL configuration to use. - * \param f_vrfy The verification callback to use during CRT verification. - * \param p_vrfy The opaque context to be passed to the callback. - */ -void mbedtls_ssl_conf_verify( mbedtls_ssl_config *conf, - int (*f_vrfy)(void *, mbedtls_x509_crt *, int, uint32_t *), - void *p_vrfy ); -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - -/** - * \brief Set the random number generator callback - * - * \param conf SSL configuration - * \param f_rng RNG function (mandatory) - * \param p_rng RNG parameter - */ -void mbedtls_ssl_conf_rng( mbedtls_ssl_config *conf, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief Set the debug callback - * - * The callback has the following argument: - * void * opaque context for the callback - * int debug level - * const char * file name - * int line number - * const char * message - * - * \param conf SSL configuration - * \param f_dbg debug function - * \param p_dbg debug parameter - */ -void mbedtls_ssl_conf_dbg( mbedtls_ssl_config *conf, - void (*f_dbg)(void *, int, const char *, int, const char *), - void *p_dbg ); - -/** - * \brief Set the underlying BIO callbacks for write, read and - * read-with-timeout. - * - * \param ssl SSL context - * \param p_bio parameter (context) shared by BIO callbacks - * \param f_send write callback - * \param f_recv read callback - * \param f_recv_timeout blocking read callback with timeout. - * - * \note One of f_recv or f_recv_timeout can be NULL, in which case - * the other is used. If both are non-NULL, f_recv_timeout is - * used and f_recv is ignored (as if it were NULL). - * - * \note The two most common use cases are: - * - non-blocking I/O, f_recv != NULL, f_recv_timeout == NULL - * - blocking I/O, f_recv == NULL, f_recv_timout != NULL - * - * \note For DTLS, you need to provide either a non-NULL - * f_recv_timeout callback, or a f_recv that doesn't block. - * - * \note See the documentations of \c mbedtls_ssl_send_t, - * \c mbedtls_ssl_recv_t and \c mbedtls_ssl_recv_timeout_t for - * the conventions those callbacks must follow. - * - * \note On some platforms, net_sockets.c provides - * \c mbedtls_net_send(), \c mbedtls_net_recv() and - * \c mbedtls_net_recv_timeout() that are suitable to be used - * here. - */ -void mbedtls_ssl_set_bio( mbedtls_ssl_context *ssl, - void *p_bio, - mbedtls_ssl_send_t *f_send, - mbedtls_ssl_recv_t *f_recv, - mbedtls_ssl_recv_timeout_t *f_recv_timeout ); - -#if defined(MBEDTLS_SSL_PROTO_DTLS) - -#if defined(MBEDTLS_SSL_DTLS_CONNECTION_ID) - - -/** - * \brief Configure the use of the Connection ID (CID) - * extension in the next handshake. - * - * Reference: draft-ietf-tls-dtls-connection-id-05 - * https://tools.ietf.org/html/draft-ietf-tls-dtls-connection-id-05 - * - * The DTLS CID extension allows the reliable association of - * DTLS records to DTLS connections across changes in the - * underlying transport (changed IP and Port metadata) by - * adding explicit connection identifiers (CIDs) to the - * headers of encrypted DTLS records. The desired CIDs are - * configured by the application layer and are exchanged in - * new `ClientHello` / `ServerHello` extensions during the - * handshake, where each side indicates the CID it wants the - * peer to use when writing encrypted messages. The CIDs are - * put to use once records get encrypted: the stack discards - * any incoming records that don't include the configured CID - * in their header, and adds the peer's requested CID to the - * headers of outgoing messages. - * - * This API enables or disables the use of the CID extension - * in the next handshake and sets the value of the CID to - * be used for incoming messages. - * - * \param ssl The SSL context to configure. This must be initialized. - * \param enable This value determines whether the CID extension should - * be used or not. Possible values are: - * - MBEDTLS_SSL_CID_ENABLED to enable the use of the CID. - * - MBEDTLS_SSL_CID_DISABLED (default) to disable the use - * of the CID. - * \param own_cid The address of the readable buffer holding the CID we want - * the peer to use when sending encrypted messages to us. - * This may be \c NULL if \p own_cid_len is \c 0. - * This parameter is unused if \p enabled is set to - * MBEDTLS_SSL_CID_DISABLED. - * \param own_cid_len The length of \p own_cid. - * This parameter is unused if \p enabled is set to - * MBEDTLS_SSL_CID_DISABLED. - * - * \note The value of \p own_cid_len must match the value of the - * \c len parameter passed to mbedtls_ssl_conf_cid() - * when configuring the ::mbedtls_ssl_config that \p ssl - * is bound to. - * - * \note This CID configuration applies to subsequent handshakes - * performed on the SSL context \p ssl, but does not trigger - * one. You still have to call `mbedtls_ssl_handshake()` - * (for the initial handshake) or `mbedtls_ssl_renegotiate()` - * (for a renegotiation handshake) explicitly after a - * successful call to this function to run the handshake. - * - * \note This call cannot guarantee that the use of the CID - * will be successfully negotiated in the next handshake, - * because the peer might not support it. Specifically: - * - On the Client, enabling the use of the CID through - * this call implies that the `ClientHello` in the next - * handshake will include the CID extension, thereby - * offering the use of the CID to the server. Only if - * the `ServerHello` contains the CID extension, too, - * the CID extension will actually be put to use. - * - On the Server, enabling the use of the CID through - * this call implies that that the server will look for - * the CID extension in a `ClientHello` from the client, - * and, if present, reply with a CID extension in its - * `ServerHello`. - * - * \note To check whether the use of the CID was negotiated - * after the subsequent handshake has completed, please - * use the API mbedtls_ssl_get_peer_cid(). - * - * \warning If the use of the CID extension is enabled in this call - * and the subsequent handshake negotiates its use, Mbed TLS - * will silently drop every packet whose CID does not match - * the CID configured in \p own_cid. It is the responsibility - * of the user to adapt the underlying transport to take care - * of CID-based demultiplexing before handing datagrams to - * Mbed TLS. - * - * \return \c 0 on success. In this case, the CID configuration - * applies to the next handshake. - * \return A negative error code on failure. - */ -int mbedtls_ssl_set_cid( mbedtls_ssl_context *ssl, - int enable, - unsigned char const *own_cid, - size_t own_cid_len ); - -/** - * \brief Get information about the use of the CID extension - * in the current connection. - * - * \param ssl The SSL context to query. - * \param enabled The address at which to store whether the CID extension - * is currently in use or not. If the CID is in use, - * `*enabled` is set to MBEDTLS_SSL_CID_ENABLED; - * otherwise, it is set to MBEDTLS_SSL_CID_DISABLED. - * \param peer_cid The address of the buffer in which to store the CID - * chosen by the peer (if the CID extension is used). - * This may be \c NULL in case the value of peer CID - * isn't needed. If it is not \c NULL, \p peer_cid_len - * must not be \c NULL. - * \param peer_cid_len The address at which to store the size of the CID - * chosen by the peer (if the CID extension is used). - * This is also the number of Bytes in \p peer_cid that - * have been written. - * This may be \c NULL in case the length of the peer CID - * isn't needed. If it is \c NULL, \p peer_cid must be - * \c NULL, too. - * - * \note This applies to the state of the CID negotiated in - * the last complete handshake. If a handshake is in - * progress, this function will attempt to complete - * the handshake first. - * - * \note If CID extensions have been exchanged but both client - * and server chose to use an empty CID, this function - * sets `*enabled` to #MBEDTLS_SSL_CID_DISABLED - * (the rationale for this is that the resulting - * communication is the same as if the CID extensions - * hadn't been used). - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -int mbedtls_ssl_get_peer_cid( mbedtls_ssl_context *ssl, - int *enabled, - unsigned char peer_cid[ MBEDTLS_SSL_CID_OUT_LEN_MAX ], - size_t *peer_cid_len ); - -#endif /* MBEDTLS_SSL_DTLS_CONNECTION_ID */ - -/** - * \brief Set the Maximum Tranport Unit (MTU). - * Special value: 0 means unset (no limit). - * This represents the maximum size of a datagram payload - * handled by the transport layer (usually UDP) as determined - * by the network link and stack. In practice, this controls - * the maximum size datagram the DTLS layer will pass to the - * \c f_send() callback set using \c mbedtls_ssl_set_bio(). - * - * \note The limit on datagram size is converted to a limit on - * record payload by subtracting the current overhead of - * encapsulation and encryption/authentication if any. - * - * \note This can be called at any point during the connection, for - * example when a Path Maximum Transfer Unit (PMTU) - * estimate becomes available from other sources, - * such as lower (or higher) protocol layers. - * - * \note This setting only controls the size of the packets we send, - * and does not restrict the size of the datagrams we're - * willing to receive. Client-side, you can request the - * server to use smaller records with \c - * mbedtls_ssl_conf_max_frag_len(). - * - * \note If both a MTU and a maximum fragment length have been - * configured (or negotiated with the peer), the resulting - * lower limit on record payload (see first note) is used. - * - * \note This can only be used to decrease the maximum size - * of datagrams (hence records, see first note) sent. It - * cannot be used to increase the maximum size of records over - * the limit set by #MBEDTLS_SSL_OUT_CONTENT_LEN. - * - * \note Values lower than the current record layer expansion will - * result in an error when trying to send data. - * - * \param ssl SSL context - * \param mtu Value of the path MTU in bytes - */ -void mbedtls_ssl_set_mtu( mbedtls_ssl_context *ssl, uint16_t mtu ); -#endif /* MBEDTLS_SSL_PROTO_DTLS */ - -#if defined(MBEDTLS_X509_CRT_PARSE_C) -/** - * \brief Set a connection-specific verification callback (optional). - * - * If set, the provided verify callback is called for each - * certificate in the peer's CRT chain, including the trusted - * root. For more information, please see the documentation of - * \c mbedtls_x509_crt_verify(). - * - * \note This call is analogous to mbedtls_ssl_conf_verify() but - * binds the verification callback and context to an SSL context - * as opposed to an SSL configuration. - * If mbedtls_ssl_conf_verify() and mbedtls_ssl_set_verify() - * are both used, mbedtls_ssl_set_verify() takes precedence. - * - * \param ssl The SSL context to use. - * \param f_vrfy The verification callback to use during CRT verification. - * \param p_vrfy The opaque context to be passed to the callback. - */ -void mbedtls_ssl_set_verify( mbedtls_ssl_context *ssl, - int (*f_vrfy)(void *, mbedtls_x509_crt *, int, uint32_t *), - void *p_vrfy ); -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - -/** - * \brief Set the timeout period for mbedtls_ssl_read() - * (Default: no timeout.) - * - * \param conf SSL configuration context - * \param timeout Timeout value in milliseconds. - * Use 0 for no timeout (default). - * - * \note With blocking I/O, this will only work if a non-NULL - * \c f_recv_timeout was set with \c mbedtls_ssl_set_bio(). - * With non-blocking I/O, this will only work if timer - * callbacks were set with \c mbedtls_ssl_set_timer_cb(). - * - * \note With non-blocking I/O, you may also skip this function - * altogether and handle timeouts at the application layer. - */ -void mbedtls_ssl_conf_read_timeout( mbedtls_ssl_config *conf, uint32_t timeout ); - -/** - * \brief Check whether a buffer contains a valid and authentic record - * that has not been seen before. (DTLS only). - * - * This function does not change the user-visible state - * of the SSL context. Its sole purpose is to provide - * an indication of the legitimacy of an incoming record. - * - * This can be useful e.g. in distributed server environments - * using the DTLS Connection ID feature, in which connections - * might need to be passed between service instances on a change - * of peer address, but where such disruptive operations should - * only happen after the validity of incoming records has been - * confirmed. - * - * \param ssl The SSL context to use. - * \param buf The address of the buffer holding the record to be checked. - * This must be a read/write buffer of length \p buflen Bytes. - * \param buflen The length of \p buf in Bytes. - * - * \note This routine only checks whether the provided buffer begins - * with a valid and authentic record that has not been seen - * before, but does not check potential data following the - * initial record. In particular, it is possible to pass DTLS - * datagrams containing multiple records, in which case only - * the first record is checked. - * - * \note This function modifies the input buffer \p buf. If you need - * to preserve the original record, you have to maintain a copy. - * - * \return \c 0 if the record is valid and authentic and has not been - * seen before. - * \return MBEDTLS_ERR_SSL_INVALID_MAC if the check completed - * successfully but the record was found to be not authentic. - * \return MBEDTLS_ERR_SSL_INVALID_RECORD if the check completed - * successfully but the record was found to be invalid for - * a reason different from authenticity checking. - * \return MBEDTLS_ERR_SSL_UNEXPECTED_RECORD if the check completed - * successfully but the record was found to be unexpected - * in the state of the SSL context, including replayed records. - * \return Another negative error code on different kinds of failure. - * In this case, the SSL context becomes unusable and needs - * to be freed or reset before reuse. - */ -int mbedtls_ssl_check_record( mbedtls_ssl_context const *ssl, - unsigned char *buf, - size_t buflen ); - -/** - * \brief Set the timer callbacks (Mandatory for DTLS.) - * - * \param ssl SSL context - * \param p_timer parameter (context) shared by timer callbacks - * \param f_set_timer set timer callback - * \param f_get_timer get timer callback. Must return: - * - * \note See the documentation of \c mbedtls_ssl_set_timer_t and - * \c mbedtls_ssl_get_timer_t for the conventions this pair of - * callbacks must follow. - * - * \note On some platforms, timing.c provides - * \c mbedtls_timing_set_delay() and - * \c mbedtls_timing_get_delay() that are suitable for using - * here, except if using an event-driven style. - * - * \note See also the "DTLS tutorial" article in our knowledge base. - * https://tls.mbed.org/kb/how-to/dtls-tutorial - */ -void mbedtls_ssl_set_timer_cb( mbedtls_ssl_context *ssl, - void *p_timer, - mbedtls_ssl_set_timer_t *f_set_timer, - mbedtls_ssl_get_timer_t *f_get_timer ); - -/** - * \brief Callback type: generate and write session ticket - * - * \note This describes what a callback implementation should do. - * This callback should generate an encrypted and - * authenticated ticket for the session and write it to the - * output buffer. Here, ticket means the opaque ticket part - * of the NewSessionTicket structure of RFC 5077. - * - * \param p_ticket Context for the callback - * \param session SSL session to be written in the ticket - * \param start Start of the output buffer - * \param end End of the output buffer - * \param tlen On exit, holds the length written - * \param lifetime On exit, holds the lifetime of the ticket in seconds - * - * \return 0 if successful, or - * a specific MBEDTLS_ERR_XXX code. - */ -typedef int mbedtls_ssl_ticket_write_t( void *p_ticket, - const mbedtls_ssl_session *session, - unsigned char *start, - const unsigned char *end, - size_t *tlen, - uint32_t *lifetime ); - -/** - * \brief Callback type: parse and load session ticket - * - * \note This describes what a callback implementation should do. - * This callback should parse a session ticket as generated - * by the corresponding mbedtls_ssl_ticket_write_t function, - * and, if the ticket is authentic and valid, load the - * session. - * - * \note The implementation is allowed to modify the first len - * bytes of the input buffer, eg to use it as a temporary - * area for the decrypted ticket contents. - * - * \param p_ticket Context for the callback - * \param session SSL session to be loaded - * \param buf Start of the buffer containing the ticket - * \param len Length of the ticket. - * - * \return 0 if successful, or - * MBEDTLS_ERR_SSL_INVALID_MAC if not authentic, or - * MBEDTLS_ERR_SSL_SESSION_TICKET_EXPIRED if expired, or - * any other non-zero code for other failures. - */ -typedef int mbedtls_ssl_ticket_parse_t( void *p_ticket, - mbedtls_ssl_session *session, - unsigned char *buf, - size_t len ); - -#if defined(MBEDTLS_SSL_SESSION_TICKETS) && defined(MBEDTLS_SSL_SRV_C) -/** - * \brief Configure SSL session ticket callbacks (server only). - * (Default: none.) - * - * \note On server, session tickets are enabled by providing - * non-NULL callbacks. - * - * \note On client, use \c mbedtls_ssl_conf_session_tickets(). - * - * \param conf SSL configuration context - * \param f_ticket_write Callback for writing a ticket - * \param f_ticket_parse Callback for parsing a ticket - * \param p_ticket Context shared by the two callbacks - */ -void mbedtls_ssl_conf_session_tickets_cb( mbedtls_ssl_config *conf, - mbedtls_ssl_ticket_write_t *f_ticket_write, - mbedtls_ssl_ticket_parse_t *f_ticket_parse, - void *p_ticket ); -#endif /* MBEDTLS_SSL_SESSION_TICKETS && MBEDTLS_SSL_SRV_C */ - -#if defined(MBEDTLS_SSL_EXPORT_KEYS) -/** - * \brief Configure a key export callback. - * (Default: none.) - * - * This API can be used for two purposes: - * - Debugging: Use this API to e.g. generate an NSSKeylog - * file and use it to inspect encrypted traffic in tools - * such as Wireshark. - * - Application-specific export: Use this API to implement - * key exporters, e.g. for EAP-TLS or DTLS-SRTP. - * - * - * \param ssl The SSL context to which the export - * callback should be attached. - * \param f_export_keys The callback for the key export. - * \param p_export_keys The opaque context pointer to be passed to the - * callback \p f_export_keys. - */ -void mbedtls_ssl_set_export_keys_cb( mbedtls_ssl_context *ssl, - mbedtls_ssl_export_keys_t *f_export_keys, - void *p_export_keys ); -#endif /* MBEDTLS_SSL_EXPORT_KEYS */ - -#if defined(MBEDTLS_SSL_ASYNC_PRIVATE) -/** - * \brief Configure asynchronous private key operation callbacks. - * - * \param conf SSL configuration context - * \param f_async_sign Callback to start a signature operation. See - * the description of ::mbedtls_ssl_async_sign_t - * for more information. This may be \c NULL if the - * external processor does not support any signature - * operation; in this case the private key object - * associated with the certificate will be used. - * \param f_async_decrypt Callback to start a decryption operation. See - * the description of ::mbedtls_ssl_async_decrypt_t - * for more information. This may be \c NULL if the - * external processor does not support any decryption - * operation; in this case the private key object - * associated with the certificate will be used. - * \param f_async_resume Callback to resume an asynchronous operation. See - * the description of ::mbedtls_ssl_async_resume_t - * for more information. This may not be \c NULL unless - * \p f_async_sign and \p f_async_decrypt are both - * \c NULL. - * \param f_async_cancel Callback to cancel an asynchronous operation. See - * the description of ::mbedtls_ssl_async_cancel_t - * for more information. This may be \c NULL if - * no cleanup is needed. - * \param config_data A pointer to configuration data which can be - * retrieved with - * mbedtls_ssl_conf_get_async_config_data(). The - * library stores this value without dereferencing it. - */ -void mbedtls_ssl_conf_async_private_cb( mbedtls_ssl_config *conf, - mbedtls_ssl_async_sign_t *f_async_sign, - mbedtls_ssl_async_decrypt_t *f_async_decrypt, - mbedtls_ssl_async_resume_t *f_async_resume, - mbedtls_ssl_async_cancel_t *f_async_cancel, - void *config_data ); - -/** - * \brief Retrieve the configuration data set by - * mbedtls_ssl_conf_async_private_cb(). - * - * \param conf SSL configuration context - * \return The configuration data set by - * mbedtls_ssl_conf_async_private_cb(). - */ -void *mbedtls_ssl_conf_get_async_config_data( const mbedtls_ssl_config *conf ); - -/** - * \brief Retrieve the asynchronous operation user context. - * - * \note This function may only be called while a handshake - * is in progress. - * - * \param ssl The SSL context to access. - * - * \return The asynchronous operation user context that was last - * set during the current handshake. If - * mbedtls_ssl_set_async_operation_data() has not yet been - * called during the current handshake, this function returns - * \c NULL. - */ -void *mbedtls_ssl_get_async_operation_data( const mbedtls_ssl_context *ssl ); - -/** - * \brief Retrieve the asynchronous operation user context. - * - * \note This function may only be called while a handshake - * is in progress. - * - * \param ssl The SSL context to access. - * \param ctx The new value of the asynchronous operation user context. - * Call mbedtls_ssl_get_async_operation_data() later during the - * same handshake to retrieve this value. - */ -void mbedtls_ssl_set_async_operation_data( mbedtls_ssl_context *ssl, - void *ctx ); -#endif /* MBEDTLS_SSL_ASYNC_PRIVATE */ - -/** - * \brief Callback type: generate a cookie - * - * \param ctx Context for the callback - * \param p Buffer to write to, - * must be updated to point right after the cookie - * \param end Pointer to one past the end of the output buffer - * \param info Client ID info that was passed to - * \c mbedtls_ssl_set_client_transport_id() - * \param ilen Length of info in bytes - * - * \return The callback must return 0 on success, - * or a negative error code. - */ -typedef int mbedtls_ssl_cookie_write_t( void *ctx, - unsigned char **p, unsigned char *end, - const unsigned char *info, size_t ilen ); - -/** - * \brief Callback type: verify a cookie - * - * \param ctx Context for the callback - * \param cookie Cookie to verify - * \param clen Length of cookie - * \param info Client ID info that was passed to - * \c mbedtls_ssl_set_client_transport_id() - * \param ilen Length of info in bytes - * - * \return The callback must return 0 if cookie is valid, - * or a negative error code. - */ -typedef int mbedtls_ssl_cookie_check_t( void *ctx, - const unsigned char *cookie, size_t clen, - const unsigned char *info, size_t ilen ); - -#if defined(MBEDTLS_SSL_DTLS_HELLO_VERIFY) && defined(MBEDTLS_SSL_SRV_C) -/** - * \brief Register callbacks for DTLS cookies - * (Server only. DTLS only.) - * - * Default: dummy callbacks that fail, in order to force you to - * register working callbacks (and initialize their context). - * - * To disable HelloVerifyRequest, register NULL callbacks. - * - * \warning Disabling hello verification allows your server to be used - * for amplification in DoS attacks against other hosts. - * Only disable if you known this can't happen in your - * particular environment. - * - * \note See comments on \c mbedtls_ssl_handshake() about handling - * the MBEDTLS_ERR_SSL_HELLO_VERIFY_REQUIRED that is expected - * on the first handshake attempt when this is enabled. - * - * \note This is also necessary to handle client reconnection from - * the same port as described in RFC 6347 section 4.2.8 (only - * the variant with cookies is supported currently). See - * comments on \c mbedtls_ssl_read() for details. - * - * \param conf SSL configuration - * \param f_cookie_write Cookie write callback - * \param f_cookie_check Cookie check callback - * \param p_cookie Context for both callbacks - */ -void mbedtls_ssl_conf_dtls_cookies( mbedtls_ssl_config *conf, - mbedtls_ssl_cookie_write_t *f_cookie_write, - mbedtls_ssl_cookie_check_t *f_cookie_check, - void *p_cookie ); - -/** - * \brief Set client's transport-level identification info. - * (Server only. DTLS only.) - * - * This is usually the IP address (and port), but could be - * anything identify the client depending on the underlying - * network stack. Used for HelloVerifyRequest with DTLS. - * This is *not* used to route the actual packets. - * - * \param ssl SSL context - * \param info Transport-level info identifying the client (eg IP + port) - * \param ilen Length of info in bytes - * - * \note An internal copy is made, so the info buffer can be reused. - * - * \return 0 on success, - * MBEDTLS_ERR_SSL_BAD_INPUT_DATA if used on client, - * MBEDTLS_ERR_SSL_ALLOC_FAILED if out of memory. - */ -int mbedtls_ssl_set_client_transport_id( mbedtls_ssl_context *ssl, - const unsigned char *info, - size_t ilen ); - -#endif /* MBEDTLS_SSL_DTLS_HELLO_VERIFY && MBEDTLS_SSL_SRV_C */ - -#if defined(MBEDTLS_SSL_DTLS_ANTI_REPLAY) -/** - * \brief Enable or disable anti-replay protection for DTLS. - * (DTLS only, no effect on TLS.) - * Default: enabled. - * - * \param conf SSL configuration - * \param mode MBEDTLS_SSL_ANTI_REPLAY_ENABLED or MBEDTLS_SSL_ANTI_REPLAY_DISABLED. - * - * \warning Disabling this is a security risk unless the application - * protocol handles duplicated packets in a safe way. You - * should not disable this without careful consideration. - * However, if your application already detects duplicated - * packets and needs information about them to adjust its - * transmission strategy, then you'll want to disable this. - */ -void mbedtls_ssl_conf_dtls_anti_replay( mbedtls_ssl_config *conf, char mode ); -#endif /* MBEDTLS_SSL_DTLS_ANTI_REPLAY */ - -/** - * \brief Set a limit on the number of records with a bad MAC - * before terminating the connection. - * (DTLS only, no effect on TLS.) - * Default: 0 (disabled). - * - * \param conf SSL configuration - * \param limit Limit, or 0 to disable. - * - * \note If the limit is N, then the connection is terminated when - * the Nth non-authentic record is seen. - * - * \note Records with an invalid header are not counted, only the - * ones going through the authentication-decryption phase. - * - * \note This is a security trade-off related to the fact that it's - * often relatively easy for an active attacker ot inject UDP - * datagrams. On one hand, setting a low limit here makes it - * easier for such an attacker to forcibly terminated a - * connection. On the other hand, a high limit or no limit - * might make us waste resources checking authentication on - * many bogus packets. - */ -void mbedtls_ssl_conf_dtls_badmac_limit( mbedtls_ssl_config *conf, unsigned limit ); - -#if defined(MBEDTLS_SSL_PROTO_DTLS) - -/** - * \brief Allow or disallow packing of multiple handshake records - * within a single datagram. - * - * \param ssl The SSL context to configure. - * \param allow_packing This determines whether datagram packing may - * be used or not. A value of \c 0 means that every - * record will be sent in a separate datagram; a - * value of \c 1 means that, if space permits, - * multiple handshake messages (including CCS) belonging to - * a single flight may be packed within a single datagram. - * - * \note This is enabled by default and should only be disabled - * for test purposes, or if datagram packing causes - * interoperability issues with peers that don't support it. - * - * \note Allowing datagram packing reduces the network load since - * there's less overhead if multiple messages share the same - * datagram. Also, it increases the handshake efficiency - * since messages belonging to a single datagram will not - * be reordered in transit, and so future message buffering - * or flight retransmission (if no buffering is used) as - * means to deal with reordering are needed less frequently. - * - * \note Application records are not affected by this option and - * are currently always sent in separate datagrams. - * - */ -void mbedtls_ssl_set_datagram_packing( mbedtls_ssl_context *ssl, - unsigned allow_packing ); - -/** - * \brief Set retransmit timeout values for the DTLS handshake. - * (DTLS only, no effect on TLS.) - * - * \param conf SSL configuration - * \param min Initial timeout value in milliseconds. - * Default: 1000 (1 second). - * \param max Maximum timeout value in milliseconds. - * Default: 60000 (60 seconds). - * - * \note Default values are from RFC 6347 section 4.2.4.1. - * - * \note The 'min' value should typically be slightly above the - * expected round-trip time to your peer, plus whatever time - * it takes for the peer to process the message. For example, - * if your RTT is about 600ms and you peer needs up to 1s to - * do the cryptographic operations in the handshake, then you - * should set 'min' slightly above 1600. Lower values of 'min' - * might cause spurious resends which waste network resources, - * while larger value of 'min' will increase overall latency - * on unreliable network links. - * - * \note The more unreliable your network connection is, the larger - * your max / min ratio needs to be in order to achieve - * reliable handshakes. - * - * \note Messages are retransmitted up to log2(ceil(max/min)) times. - * For example, if min = 1s and max = 5s, the retransmit plan - * goes: send ... 1s -> resend ... 2s -> resend ... 4s -> - * resend ... 5s -> give up and return a timeout error. - */ -void mbedtls_ssl_conf_handshake_timeout( mbedtls_ssl_config *conf, uint32_t min, uint32_t max ); -#endif /* MBEDTLS_SSL_PROTO_DTLS */ - -#if defined(MBEDTLS_SSL_SRV_C) -/** - * \brief Set the session cache callbacks (server-side only) - * If not set, no session resuming is done (except if session - * tickets are enabled too). - * - * The session cache has the responsibility to check for stale - * entries based on timeout. See RFC 5246 for recommendations. - * - * Warning: session.peer_cert is cleared by the SSL/TLS layer on - * connection shutdown, so do not cache the pointer! Either set - * it to NULL or make a full copy of the certificate. - * - * The get callback is called once during the initial handshake - * to enable session resuming. The get function has the - * following parameters: (void *parameter, mbedtls_ssl_session *session) - * If a valid entry is found, it should fill the master of - * the session object with the cached values and return 0, - * return 1 otherwise. Optionally peer_cert can be set as well - * if it is properly present in cache entry. - * - * The set callback is called once during the initial handshake - * to enable session resuming after the entire handshake has - * been finished. The set function has the following parameters: - * (void *parameter, const mbedtls_ssl_session *session). The function - * should create a cache entry for future retrieval based on - * the data in the session structure and should keep in mind - * that the mbedtls_ssl_session object presented (and all its referenced - * data) is cleared by the SSL/TLS layer when the connection is - * terminated. It is recommended to add metadata to determine if - * an entry is still valid in the future. Return 0 if - * successfully cached, return 1 otherwise. - * - * \param conf SSL configuration - * \param p_cache parmater (context) for both callbacks - * \param f_get_cache session get callback - * \param f_set_cache session set callback - */ -void mbedtls_ssl_conf_session_cache( mbedtls_ssl_config *conf, - void *p_cache, - mbedtls_ssl_cache_get_t *f_get_cache, - mbedtls_ssl_cache_set_t *f_set_cache ); -#endif /* MBEDTLS_SSL_SRV_C */ - -#if defined(MBEDTLS_SSL_CLI_C) -/** - * \brief Load a session for session resumption. - * - * Sessions loaded through this call will be considered - * for session resumption in the next handshake. - * - * \note Even if this call succeeds, it is not guaranteed that - * the next handshake will indeed be shortened through the - * use of session resumption: The server is always free - * to reject any attempt for resumption and fall back to - * a full handshake. - * - * \note This function can handle a variety of mechanisms for session - * resumption: For TLS 1.2, both session ID-based resumption and - * ticket-based resumption will be considered. For TLS 1.3, - * once implemented, sessions equate to tickets, and loading - * one or more sessions via this call will lead to their - * corresponding tickets being advertised as resumption PSKs - * by the client. - * - * \note Calling this function multiple times will only be useful - * once TLS 1.3 is supported. For TLS 1.2 connections, this - * function should be called at most once. - * - * \param ssl The SSL context representing the connection which should - * be attempted to be setup using session resumption. This - * must be initialized via mbedtls_ssl_init() and bound to - * an SSL configuration via mbedtls_ssl_setup(), but - * the handshake must not yet have been started. - * \param session The session to be considered for session resumption. - * This must be a session previously exported via - * mbedtls_ssl_get_session(), and potentially serialized and - * deserialized through mbedtls_ssl_session_save() and - * mbedtls_ssl_session_load() in the meantime. - * - * \return \c 0 if successful. - * \return \c MBEDTLS_ERR_SSL_FEATURE_UNAVAILABLE if the session - * could not be loaded because of an implementation limitation. - * This error is non-fatal, and has no observable effect on - * the SSL context or the session that was attempted to be loaded. - * \return Another negative error code on other kinds of failure. - * - * \sa mbedtls_ssl_get_session() - * \sa mbedtls_ssl_session_load() - */ -int mbedtls_ssl_set_session( mbedtls_ssl_context *ssl, const mbedtls_ssl_session *session ); -#endif /* MBEDTLS_SSL_CLI_C */ - -/** - * \brief Load serialized session data into a session structure. - * On client, this can be used for loading saved sessions - * before resuming them with mbedstls_ssl_set_session(). - * On server, this can be used for alternative implementations - * of session cache or session tickets. - * - * \warning If a peer certificate chain is associated with the session, - * the serialized state will only contain the peer's - * end-entity certificate and the result of the chain - * verification (unless verification was disabled), but not - * the rest of the chain. - * - * \see mbedtls_ssl_session_save() - * \see mbedtls_ssl_set_session() - * - * \param session The session structure to be populated. It must have been - * initialised with mbedtls_ssl_session_init() but not - * populated yet. - * \param buf The buffer holding the serialized session data. It must be a - * readable buffer of at least \p len bytes. - * \param len The size of the serialized data in bytes. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_SSL_ALLOC_FAILED if memory allocation failed. - * \return #MBEDTLS_ERR_SSL_BAD_INPUT_DATA if input data is invalid. - * \return #MBEDTLS_ERR_SSL_VERSION_MISMATCH if the serialized data - * was generated in a different version or configuration of - * Mbed TLS. - * \return Another negative value for other kinds of errors (for - * example, unsupported features in the embedded certificate). - */ -int mbedtls_ssl_session_load( mbedtls_ssl_session *session, - const unsigned char *buf, - size_t len ); - -/** - * \brief Save session structure as serialized data in a buffer. - * On client, this can be used for saving session data, - * potentially in non-volatile storage, for resuming later. - * On server, this can be used for alternative implementations - * of session cache or session tickets. - * - * \see mbedtls_ssl_session_load() - * - * \param session The session structure to be saved. - * \param buf The buffer to write the serialized data to. It must be a - * writeable buffer of at least \p len bytes, or may be \c - * NULL if \p len is \c 0. - * \param buf_len The number of bytes available for writing in \p buf. - * \param olen The size in bytes of the data that has been or would have - * been written. It must point to a valid \c size_t. - * - * \note \p olen is updated to the correct value regardless of - * whether \p buf_len was large enough. This makes it possible - * to determine the necessary size by calling this function - * with \p buf set to \c NULL and \p buf_len to \c 0. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_SSL_BUFFER_TOO_SMALL if \p buf is too small. - */ -int mbedtls_ssl_session_save( const mbedtls_ssl_session *session, - unsigned char *buf, - size_t buf_len, - size_t *olen ); - -/** - * \brief Set the list of allowed ciphersuites and the preference - * order. First in the list has the highest preference. - * (Overrides all version-specific lists) - * - * The ciphersuites array is not copied, and must remain - * valid for the lifetime of the ssl_config. - * - * Note: By default, the server chooses its preferred - * ciphersuite among those that the client supports. If - * mbedtls_ssl_conf_preference_order() is called to prefer - * the client's preferences, the server instead chooses - * the client's preferred ciphersuite among those that - * the server supports. - * - * \param conf SSL configuration - * \param ciphersuites 0-terminated list of allowed ciphersuites - */ -void mbedtls_ssl_conf_ciphersuites( mbedtls_ssl_config *conf, - const int *ciphersuites ); - -#if defined(MBEDTLS_SSL_DTLS_CONNECTION_ID) -#define MBEDTLS_SSL_UNEXPECTED_CID_IGNORE 0 -#define MBEDTLS_SSL_UNEXPECTED_CID_FAIL 1 -/** - * \brief Specify the length of Connection IDs for incoming - * encrypted DTLS records, as well as the behaviour - * on unexpected CIDs. - * - * By default, the CID length is set to \c 0, - * and unexpected CIDs are silently ignored. - * - * \param conf The SSL configuration to modify. - * \param len The length in Bytes of the CID fields in encrypted - * DTLS records using the CID mechanism. This must - * not be larger than #MBEDTLS_SSL_CID_OUT_LEN_MAX. - * \param ignore_other_cids This determines the stack's behaviour when - * receiving a record with an unexpected CID. - * Possible values are: - * - #MBEDTLS_SSL_UNEXPECTED_CID_IGNORE - * In this case, the record is silently ignored. - * - #MBEDTLS_SSL_UNEXPECTED_CID_FAIL - * In this case, the stack fails with the specific - * error code #MBEDTLS_ERR_SSL_UNEXPECTED_CID. - * - * \note The CID specification allows implementations to either - * use a common length for all incoming connection IDs or - * allow variable-length incoming IDs. Mbed TLS currently - * requires a common length for all connections sharing the - * same SSL configuration; this allows simpler parsing of - * record headers. - * - * \return \c 0 on success. - * \return #MBEDTLS_ERR_SSL_BAD_INPUT_DATA if \p own_cid_len - * is too large. - */ -int mbedtls_ssl_conf_cid( mbedtls_ssl_config *conf, size_t len, - int ignore_other_cids ); -#endif /* MBEDTLS_SSL_DTLS_CONNECTION_ID */ - -#if defined(MBEDTLS_X509_CRT_PARSE_C) -/** - * \brief Set the X.509 security profile used for verification - * - * \note The restrictions are enforced for all certificates in the - * chain. However, signatures in the handshake are not covered - * by this setting but by \b mbedtls_ssl_conf_sig_hashes(). - * - * \param conf SSL configuration - * \param profile Profile to use - */ -void mbedtls_ssl_conf_cert_profile( mbedtls_ssl_config *conf, - const mbedtls_x509_crt_profile *profile ); - -/** - * \brief Set the data required to verify peer certificate - * - * \note See \c mbedtls_x509_crt_verify() for notes regarding the - * parameters ca_chain (maps to trust_ca for that function) - * and ca_crl. - * - * \param conf SSL configuration - * \param ca_chain trusted CA chain (meaning all fully trusted top-level CAs) - * \param ca_crl trusted CA CRLs - */ -void mbedtls_ssl_conf_ca_chain( mbedtls_ssl_config *conf, - mbedtls_x509_crt *ca_chain, - mbedtls_x509_crl *ca_crl ); - -#if defined(MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK) -/** - * \brief Set the trusted certificate callback. - * - * This API allows to register the set of trusted certificates - * through a callback, instead of a linked list as configured - * by mbedtls_ssl_conf_ca_chain(). - * - * This is useful for example in contexts where a large number - * of CAs are used, and the inefficiency of maintaining them - * in a linked list cannot be tolerated. It is also useful when - * the set of trusted CAs needs to be modified frequently. - * - * See the documentation of `mbedtls_x509_crt_ca_cb_t` for - * more information. - * - * \param conf The SSL configuration to register the callback with. - * \param f_ca_cb The trusted certificate callback to use when verifying - * certificate chains. - * \param p_ca_cb The context to be passed to \p f_ca_cb (for example, - * a reference to a trusted CA database). - * - * \note This API is incompatible with mbedtls_ssl_conf_ca_chain(): - * Any call to this function overwrites the values set through - * earlier calls to mbedtls_ssl_conf_ca_chain() or - * mbedtls_ssl_conf_ca_cb(). - * - * \note This API is incompatible with CA indication in - * CertificateRequest messages: A server-side SSL context which - * is bound to an SSL configuration that uses a CA callback - * configured via mbedtls_ssl_conf_ca_cb(), and which requires - * client authentication, will send an empty CA list in the - * corresponding CertificateRequest message. - * - * \note This API is incompatible with mbedtls_ssl_set_hs_ca_chain(): - * If an SSL context is bound to an SSL configuration which uses - * CA callbacks configured via mbedtls_ssl_conf_ca_cb(), then - * calls to mbedtls_ssl_set_hs_ca_chain() have no effect. - * - * \note The use of this API disables the use of restartable ECC - * during X.509 CRT signature verification (but doesn't affect - * other uses). - * - * \warning This API is incompatible with the use of CRLs. Any call to - * mbedtls_ssl_conf_ca_cb() unsets CRLs configured through - * earlier calls to mbedtls_ssl_conf_ca_chain(). - * - * \warning In multi-threaded environments, the callback \p f_ca_cb - * must be thread-safe, and it is the user's responsibility - * to guarantee this (for example through a mutex - * contained in the callback context pointed to by \p p_ca_cb). - */ -void mbedtls_ssl_conf_ca_cb( mbedtls_ssl_config *conf, - mbedtls_x509_crt_ca_cb_t f_ca_cb, - void *p_ca_cb ); -#endif /* MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK */ - -/** - * \brief Set own certificate chain and private key - * - * \note own_cert should contain in order from the bottom up your - * certificate chain. The top certificate (self-signed) - * can be omitted. - * - * \note On server, this function can be called multiple times to - * provision more than one cert/key pair (eg one ECDSA, one - * RSA with SHA-256, one RSA with SHA-1). An adequate - * certificate will be selected according to the client's - * advertised capabilities. In case multiple certificates are - * adequate, preference is given to the one set by the first - * call to this function, then second, etc. - * - * \note On client, only the first call has any effect. That is, - * only one client certificate can be provisioned. The - * server's preferences in its CertficateRequest message will - * be ignored and our only cert will be sent regardless of - * whether it matches those preferences - the server can then - * decide what it wants to do with it. - * - * \note The provided \p pk_key needs to match the public key in the - * first certificate in \p own_cert, or all handshakes using - * that certificate will fail. It is your responsibility - * to ensure that; this function will not perform any check. - * You may use mbedtls_pk_check_pair() in order to perform - * this check yourself, but be aware that this function can - * be computationally expensive on some key types. - * - * \param conf SSL configuration - * \param own_cert own public certificate chain - * \param pk_key own private key - * - * \return 0 on success or MBEDTLS_ERR_SSL_ALLOC_FAILED - */ -int mbedtls_ssl_conf_own_cert( mbedtls_ssl_config *conf, - mbedtls_x509_crt *own_cert, - mbedtls_pk_context *pk_key ); -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - -#if defined(MBEDTLS_KEY_EXCHANGE_SOME_PSK_ENABLED) -/** - * \brief Configure pre-shared keys (PSKs) and their - * identities to be used in PSK-based ciphersuites. - * - * Only one PSK can be registered, through either - * mbedtls_ssl_conf_psk() or mbedtls_ssl_conf_psk_opaque(). - * If you attempt to register more than one PSK, this function - * fails, though this may change in future versions, which - * may add support for multiple PSKs. - * - * \note This is mainly useful for clients. Servers will usually - * want to use \c mbedtls_ssl_conf_psk_cb() instead. - * - * \note A PSK set by \c mbedtls_ssl_set_hs_psk() in the PSK callback - * takes precedence over a PSK configured by this function. - * - * \param conf The SSL configuration to register the PSK with. - * \param psk The pointer to the pre-shared key to use. - * \param psk_len The length of the pre-shared key in bytes. - * \param psk_identity The pointer to the pre-shared key identity. - * \param psk_identity_len The length of the pre-shared key identity - * in bytes. - * - * \note The PSK and its identity are copied internally and - * hence need not be preserved by the caller for the lifetime - * of the SSL configuration. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_SSL_FEATURE_UNAVAILABLE if no more PSKs - * can be configured. In this case, the old PSK(s) remain intact. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ssl_conf_psk( mbedtls_ssl_config *conf, - const unsigned char *psk, size_t psk_len, - const unsigned char *psk_identity, size_t psk_identity_len ); - -#if defined(MBEDTLS_USE_PSA_CRYPTO) -/** - * \brief Configure one or more opaque pre-shared keys (PSKs) and - * their identities to be used in PSK-based ciphersuites. - * - * Only one PSK can be registered, through either - * mbedtls_ssl_conf_psk() or mbedtls_ssl_conf_psk_opaque(). - * If you attempt to register more than one PSK, this function - * fails, though this may change in future versions, which - * may add support for multiple PSKs. - * - * \note This is mainly useful for clients. Servers will usually - * want to use \c mbedtls_ssl_conf_psk_cb() instead. - * - * \note An opaque PSK set by \c mbedtls_ssl_set_hs_psk_opaque() in - * the PSK callback takes precedence over an opaque PSK - * configured by this function. - * - * \param conf The SSL configuration to register the PSK with. - * \param psk The identifier of the key slot holding the PSK. - * Until \p conf is destroyed or this function is successfully - * called again, the key slot \p psk must be populated with a - * key of type PSA_ALG_CATEGORY_KEY_DERIVATION whose policy - * allows its use for the key derivation algorithm applied - * in the handshake. - * \param psk_identity The pointer to the pre-shared key identity. - * \param psk_identity_len The length of the pre-shared key identity - * in bytes. - * - * \note The PSK identity hint is copied internally and hence need - * not be preserved by the caller for the lifetime of the - * SSL configuration. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_SSL_FEATURE_UNAVAILABLE if no more PSKs - * can be configured. In this case, the old PSK(s) remain intact. - * \return Another negative error code on other kinds of failure. - */ -int mbedtls_ssl_conf_psk_opaque( mbedtls_ssl_config *conf, - psa_key_id_t psk, - const unsigned char *psk_identity, - size_t psk_identity_len ); -#endif /* MBEDTLS_USE_PSA_CRYPTO */ - -/** - * \brief Set the pre-shared Key (PSK) for the current handshake. - * - * \note This should only be called inside the PSK callback, - * i.e. the function passed to \c mbedtls_ssl_conf_psk_cb(). - * - * \note A PSK set by this function takes precedence over a PSK - * configured by \c mbedtls_ssl_conf_psk(). - * - * \param ssl The SSL context to configure a PSK for. - * \param psk The pointer to the pre-shared key. - * \param psk_len The length of the pre-shared key in bytes. - * - * \return \c 0 if successful. - * \return An \c MBEDTLS_ERR_SSL_XXX error code on failure. - */ -int mbedtls_ssl_set_hs_psk( mbedtls_ssl_context *ssl, - const unsigned char *psk, size_t psk_len ); - -#if defined(MBEDTLS_USE_PSA_CRYPTO) -/** - * \brief Set an opaque pre-shared Key (PSK) for the current handshake. - * - * \note This should only be called inside the PSK callback, - * i.e. the function passed to \c mbedtls_ssl_conf_psk_cb(). - * - * \note An opaque PSK set by this function takes precedence over an - * opaque PSK configured by \c mbedtls_ssl_conf_psk_opaque(). - * - * \param ssl The SSL context to configure a PSK for. - * \param psk The identifier of the key slot holding the PSK. - * For the duration of the current handshake, the key slot - * must be populated with a key of type - * PSA_ALG_CATEGORY_KEY_DERIVATION whose policy allows its - * use for the key derivation algorithm - * applied in the handshake. - * - * \return \c 0 if successful. - * \return An \c MBEDTLS_ERR_SSL_XXX error code on failure. - */ -int mbedtls_ssl_set_hs_psk_opaque( mbedtls_ssl_context *ssl, - psa_key_id_t psk ); -#endif /* MBEDTLS_USE_PSA_CRYPTO */ - -/** - * \brief Set the PSK callback (server-side only). - * - * If set, the PSK callback is called for each - * handshake where a PSK-based ciphersuite was negotiated. - * The caller provides the identity received and wants to - * receive the actual PSK data and length. - * - * The callback has the following parameters: - * - \c void*: The opaque pointer \p p_psk. - * - \c mbedtls_ssl_context*: The SSL context to which - * the operation applies. - * - \c const unsigned char*: The PSK identity - * selected by the client. - * - \c size_t: The length of the PSK identity - * selected by the client. - * - * If a valid PSK identity is found, the callback should use - * \c mbedtls_ssl_set_hs_psk() or - * \c mbedtls_ssl_set_hs_psk_opaque() - * on the SSL context to set the correct PSK and return \c 0. - * Any other return value will result in a denied PSK identity. - * - * \note A dynamic PSK (i.e. set by the PSK callback) takes - * precedence over a static PSK (i.e. set by - * \c mbedtls_ssl_conf_psk() or - * \c mbedtls_ssl_conf_psk_opaque()). - * This means that if you set a PSK callback using this - * function, you don't need to set a PSK using - * \c mbedtls_ssl_conf_psk() or - * \c mbedtls_ssl_conf_psk_opaque()). - * - * \param conf The SSL configuration to register the callback with. - * \param f_psk The callback for selecting and setting the PSK based - * in the PSK identity chosen by the client. - * \param p_psk A pointer to an opaque structure to be passed to - * the callback, for example a PSK store. - */ -void mbedtls_ssl_conf_psk_cb( mbedtls_ssl_config *conf, - int (*f_psk)(void *, mbedtls_ssl_context *, const unsigned char *, - size_t), - void *p_psk ); -#endif /* MBEDTLS_KEY_EXCHANGE_SOME_PSK_ENABLED */ - -#if defined(MBEDTLS_DHM_C) && defined(MBEDTLS_SSL_SRV_C) -/** - * \brief Set the Diffie-Hellman public P and G values - * from big-endian binary presentations. - * (Default values: MBEDTLS_DHM_RFC3526_MODP_2048_[PG]_BIN) - * - * \param conf SSL configuration - * \param dhm_P Diffie-Hellman-Merkle modulus in big-endian binary form - * \param P_len Length of DHM modulus - * \param dhm_G Diffie-Hellman-Merkle generator in big-endian binary form - * \param G_len Length of DHM generator - * - * \return 0 if successful - */ -int mbedtls_ssl_conf_dh_param_bin( mbedtls_ssl_config *conf, - const unsigned char *dhm_P, size_t P_len, - const unsigned char *dhm_G, size_t G_len ); - -/** - * \brief Set the Diffie-Hellman public P and G values, - * read from existing context (server-side only) - * - * \param conf SSL configuration - * \param dhm_ctx Diffie-Hellman-Merkle context - * - * \return 0 if successful - */ -int mbedtls_ssl_conf_dh_param_ctx( mbedtls_ssl_config *conf, mbedtls_dhm_context *dhm_ctx ); -#endif /* MBEDTLS_DHM_C && defined(MBEDTLS_SSL_SRV_C) */ - -#if defined(MBEDTLS_DHM_C) && defined(MBEDTLS_SSL_CLI_C) -/** - * \brief Set the minimum length for Diffie-Hellman parameters. - * (Client-side only.) - * (Default: 1024 bits.) - * - * \param conf SSL configuration - * \param bitlen Minimum bit length of the DHM prime - */ -void mbedtls_ssl_conf_dhm_min_bitlen( mbedtls_ssl_config *conf, - unsigned int bitlen ); -#endif /* MBEDTLS_DHM_C && MBEDTLS_SSL_CLI_C */ - -#if defined(MBEDTLS_ECP_C) -/** - * \brief Set the allowed curves in order of preference. - * - * On server: this only affects selection of the ECDHE curve; - * the curves used for ECDH and ECDSA are determined by the - * list of available certificates instead. - * - * On client: this affects the list of curves offered for any - * use. The server can override our preference order. - * - * Both sides: limits the set of curves accepted for use in - * ECDHE and in the peer's end-entity certificate. - * - * \note This has no influence on which curves are allowed inside the - * certificate chains, see \c mbedtls_ssl_conf_cert_profile() - * for that. For the end-entity certificate however, the key - * will be accepted only if it is allowed both by this list - * and by the cert profile. - * - * \note This list should be ordered by decreasing preference - * (preferred curve first). - * - * \note The default list is the same set of curves that - * #mbedtls_x509_crt_profile_default allows, plus - * ECDHE-only curves selected according to the same criteria. - * The order favors curves with the lowest resource usage. - * - * \note New minor versions of Mbed TLS may extend this list, - * for example if new curves are added to the library. - * New minor versions of Mbed TLS will not remove items - * from this list unless serious security concerns require it. - * New minor versions of Mbed TLS may change the order in - * keeping with the general principle of favoring the lowest - * resource usage. - * - * \param conf SSL configuration - * \param curves Ordered list of allowed curves, - * terminated by MBEDTLS_ECP_DP_NONE. - */ -void mbedtls_ssl_conf_curves( mbedtls_ssl_config *conf, - const mbedtls_ecp_group_id *curves ); -#endif /* MBEDTLS_ECP_C */ - -#if defined(MBEDTLS_KEY_EXCHANGE_WITH_CERT_ENABLED) -/** - * \brief Set the allowed hashes for signatures during the handshake. - * - * \note This only affects which hashes are offered and can be used - * for signatures during the handshake. Hashes for message - * authentication and the TLS PRF are controlled by the - * ciphersuite, see \c mbedtls_ssl_conf_ciphersuites(). Hashes - * used for certificate signature are controlled by the - * verification profile, see \c mbedtls_ssl_conf_cert_profile(). - * - * \note This list should be ordered by decreasing preference - * (preferred hash first). - * - * \note By default, all supported hashes whose length is at least - * 256 bits are allowed. This is the same set as the default - * for certificate verification - * (#mbedtls_x509_crt_profile_default). - * The preference order is currently unspecified and may - * change in future versions. - * - * \note New minor versions of Mbed TLS may extend this list, - * for example if new curves are added to the library. - * New minor versions of Mbed TLS will not remove items - * from this list unless serious security concerns require it. - * - * \param conf SSL configuration - * \param hashes Ordered list of allowed signature hashes, - * terminated by \c MBEDTLS_MD_NONE. - */ -void mbedtls_ssl_conf_sig_hashes( mbedtls_ssl_config *conf, - const int *hashes ); -#endif /* MBEDTLS_KEY_EXCHANGE_WITH_CERT_ENABLED */ - -#if defined(MBEDTLS_X509_CRT_PARSE_C) -/** - * \brief Set or reset the hostname to check against the received - * server certificate. It sets the ServerName TLS extension, - * too, if that extension is enabled. (client-side only) - * - * \param ssl SSL context - * \param hostname the server hostname, may be NULL to clear hostname - - * \note Maximum hostname length MBEDTLS_SSL_MAX_HOST_NAME_LEN. - * - * \return 0 if successful, MBEDTLS_ERR_SSL_ALLOC_FAILED on - * allocation failure, MBEDTLS_ERR_SSL_BAD_INPUT_DATA on - * too long input hostname. - * - * Hostname set to the one provided on success (cleared - * when NULL). On allocation failure hostname is cleared. - * On too long input failure, old hostname is unchanged. - */ -int mbedtls_ssl_set_hostname( mbedtls_ssl_context *ssl, const char *hostname ); -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - -#if defined(MBEDTLS_SSL_SERVER_NAME_INDICATION) -/** - * \brief Set own certificate and key for the current handshake - * - * \note Same as \c mbedtls_ssl_conf_own_cert() but for use within - * the SNI callback. - * - * \param ssl SSL context - * \param own_cert own public certificate chain - * \param pk_key own private key - * - * \return 0 on success or MBEDTLS_ERR_SSL_ALLOC_FAILED - */ -int mbedtls_ssl_set_hs_own_cert( mbedtls_ssl_context *ssl, - mbedtls_x509_crt *own_cert, - mbedtls_pk_context *pk_key ); - -/** - * \brief Set the data required to verify peer certificate for the - * current handshake - * - * \note Same as \c mbedtls_ssl_conf_ca_chain() but for use within - * the SNI callback. - * - * \param ssl SSL context - * \param ca_chain trusted CA chain (meaning all fully trusted top-level CAs) - * \param ca_crl trusted CA CRLs - */ -void mbedtls_ssl_set_hs_ca_chain( mbedtls_ssl_context *ssl, - mbedtls_x509_crt *ca_chain, - mbedtls_x509_crl *ca_crl ); - -/** - * \brief Set authmode for the current handshake. - * - * \note Same as \c mbedtls_ssl_conf_authmode() but for use within - * the SNI callback. - * - * \param ssl SSL context - * \param authmode MBEDTLS_SSL_VERIFY_NONE, MBEDTLS_SSL_VERIFY_OPTIONAL or - * MBEDTLS_SSL_VERIFY_REQUIRED - */ -void mbedtls_ssl_set_hs_authmode( mbedtls_ssl_context *ssl, - int authmode ); - -/** - * \brief Set server side ServerName TLS extension callback - * (optional, server-side only). - * - * If set, the ServerName callback is called whenever the - * server receives a ServerName TLS extension from the client - * during a handshake. The ServerName callback has the - * following parameters: (void *parameter, mbedtls_ssl_context *ssl, - * const unsigned char *hostname, size_t len). If a suitable - * certificate is found, the callback must set the - * certificate(s) and key(s) to use with \c - * mbedtls_ssl_set_hs_own_cert() (can be called repeatedly), - * and may optionally adjust the CA and associated CRL with \c - * mbedtls_ssl_set_hs_ca_chain() as well as the client - * authentication mode with \c mbedtls_ssl_set_hs_authmode(), - * then must return 0. If no matching name is found, the - * callback must either set a default cert, or - * return non-zero to abort the handshake at this point. - * - * \param conf SSL configuration - * \param f_sni verification function - * \param p_sni verification parameter - */ -void mbedtls_ssl_conf_sni( mbedtls_ssl_config *conf, - int (*f_sni)(void *, mbedtls_ssl_context *, const unsigned char *, - size_t), - void *p_sni ); -#endif /* MBEDTLS_SSL_SERVER_NAME_INDICATION */ - -#if defined(MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED) -/** - * \brief Set the EC J-PAKE password for current handshake. - * - * \note An internal copy is made, and destroyed as soon as the - * handshake is completed, or when the SSL context is reset or - * freed. - * - * \note The SSL context needs to be already set up. The right place - * to call this function is between \c mbedtls_ssl_setup() or - * \c mbedtls_ssl_reset() and \c mbedtls_ssl_handshake(). - * - * \param ssl SSL context - * \param pw EC J-PAKE password (pre-shared secret) - * \param pw_len length of pw in bytes - * - * \return 0 on success, or a negative error code. - */ -int mbedtls_ssl_set_hs_ecjpake_password( mbedtls_ssl_context *ssl, - const unsigned char *pw, - size_t pw_len ); -#endif /*MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED */ - -#if defined(MBEDTLS_SSL_ALPN) -/** - * \brief Set the supported Application Layer Protocols. - * - * \param conf SSL configuration - * \param protos Pointer to a NULL-terminated list of supported protocols, - * in decreasing preference order. The pointer to the list is - * recorded by the library for later reference as required, so - * the lifetime of the table must be atleast as long as the - * lifetime of the SSL configuration structure. - * - * \return 0 on success, or MBEDTLS_ERR_SSL_BAD_INPUT_DATA. - */ -int mbedtls_ssl_conf_alpn_protocols( mbedtls_ssl_config *conf, const char **protos ); - -/** - * \brief Get the name of the negotiated Application Layer Protocol. - * This function should be called after the handshake is - * completed. - * - * \param ssl SSL context - * - * \return Protcol name, or NULL if no protocol was negotiated. - */ -const char *mbedtls_ssl_get_alpn_protocol( const mbedtls_ssl_context *ssl ); -#endif /* MBEDTLS_SSL_ALPN */ - -#if defined(MBEDTLS_SSL_DTLS_SRTP) -#if defined(MBEDTLS_DEBUG_C) -static inline const char *mbedtls_ssl_get_srtp_profile_as_string( mbedtls_ssl_srtp_profile profile ) -{ - switch( profile ) - { - case MBEDTLS_TLS_SRTP_AES128_CM_HMAC_SHA1_80: - return( "MBEDTLS_TLS_SRTP_AES128_CM_HMAC_SHA1_80" ); - case MBEDTLS_TLS_SRTP_AES128_CM_HMAC_SHA1_32: - return( "MBEDTLS_TLS_SRTP_AES128_CM_HMAC_SHA1_32" ); - case MBEDTLS_TLS_SRTP_NULL_HMAC_SHA1_80: - return( "MBEDTLS_TLS_SRTP_NULL_HMAC_SHA1_80" ); - case MBEDTLS_TLS_SRTP_NULL_HMAC_SHA1_32: - return( "MBEDTLS_TLS_SRTP_NULL_HMAC_SHA1_32" ); - default: break; - } - return( "" ); -} -#endif /* MBEDTLS_DEBUG_C */ -/** - * \brief Manage support for mki(master key id) value - * in use_srtp extension. - * MKI is an optional part of SRTP used for key management - * and re-keying. See RFC3711 section 3.1 for details. - * The default value is - * #MBEDTLS_SSL_DTLS_SRTP_MKI_UNSUPPORTED. - * - * \param conf The SSL configuration to manage mki support. - * \param support_mki_value Enable or disable mki usage. Values are - * #MBEDTLS_SSL_DTLS_SRTP_MKI_UNSUPPORTED - * or #MBEDTLS_SSL_DTLS_SRTP_MKI_SUPPORTED. - */ -void mbedtls_ssl_conf_srtp_mki_value_supported( mbedtls_ssl_config *conf, - int support_mki_value ); - -/** - * \brief Set the supported DTLS-SRTP protection profiles. - * - * \param conf SSL configuration - * \param profiles Pointer to a List of MBEDTLS_TLS_SRTP_UNSET terminated - * supported protection profiles - * in decreasing preference order. - * The pointer to the list is recorded by the library - * for later reference as required, so the lifetime - * of the table must be at least as long as the lifetime - * of the SSL configuration structure. - * The list must not hold more than - * MBEDTLS_TLS_SRTP_MAX_PROFILE_LIST_LENGTH elements - * (excluding the terminating MBEDTLS_TLS_SRTP_UNSET). - * - * \return 0 on success - * \return #MBEDTLS_ERR_SSL_BAD_INPUT_DATA when the list of - * protection profiles is incorrect. - */ -int mbedtls_ssl_conf_dtls_srtp_protection_profiles - ( mbedtls_ssl_config *conf, - const mbedtls_ssl_srtp_profile *profiles ); - -/** - * \brief Set the mki_value for the current DTLS-SRTP session. - * - * \param ssl SSL context to use. - * \param mki_value The MKI value to set. - * \param mki_len The length of the MKI value. - * - * \note This function is relevant on client side only. - * The server discovers the mki value during handshake. - * A mki value set on server side using this function - * is ignored. - * - * \return 0 on success - * \return #MBEDTLS_ERR_SSL_BAD_INPUT_DATA - * \return #MBEDTLS_ERR_SSL_FEATURE_UNAVAILABLE - */ -int mbedtls_ssl_dtls_srtp_set_mki_value( mbedtls_ssl_context *ssl, - unsigned char *mki_value, - uint16_t mki_len ); -/** - * \brief Get the negotiated DTLS-SRTP informations: - * Protection profile and MKI value. - * - * \warning This function must be called after the handshake is - * completed. The value returned by this function must - * not be trusted or acted upon before the handshake completes. - * - * \param ssl The SSL context to query. - * \param dtls_srtp_info The negotiated DTLS-SRTP informations: - * - Protection profile in use. - * A direct mapping of the iana defined value for protection - * profile on an uint16_t. - http://www.iana.org/assignments/srtp-protection/srtp-protection.xhtml - * #MBEDTLS_TLS_SRTP_UNSET if the use of SRTP was not negotiated - * or peer's Hello packet was not parsed yet. - * - mki size and value( if size is > 0 ). - */ -void mbedtls_ssl_get_dtls_srtp_negotiation_result( const mbedtls_ssl_context *ssl, - mbedtls_dtls_srtp_info *dtls_srtp_info ); -#endif /* MBEDTLS_SSL_DTLS_SRTP */ - -/** - * \brief Set the maximum supported version sent from the client side - * and/or accepted at the server side - * (Default: MBEDTLS_SSL_MAX_MAJOR_VERSION, MBEDTLS_SSL_MAX_MINOR_VERSION) - * - * \note This ignores ciphersuites from higher versions. - * - * \note With DTLS, use MBEDTLS_SSL_MINOR_VERSION_3 for DTLS 1.2 - * - * \param conf SSL configuration - * \param major Major version number (only MBEDTLS_SSL_MAJOR_VERSION_3 supported) - * \param minor Minor version number (only MBEDTLS_SSL_MINOR_VERSION_3 supported) - */ -void mbedtls_ssl_conf_max_version( mbedtls_ssl_config *conf, int major, int minor ); - -/** - * \brief Set the minimum accepted SSL/TLS protocol version - * (Default: TLS 1.2) - * - * \note Input outside of the SSL_MAX_XXXXX_VERSION and - * SSL_MIN_XXXXX_VERSION range is ignored. - * - * \note With DTLS, use MBEDTLS_SSL_MINOR_VERSION_3 for DTLS 1.2 - * - * \param conf SSL configuration - * \param major Major version number (only MBEDTLS_SSL_MAJOR_VERSION_3 supported) - * \param minor Minor version number (only MBEDTLS_SSL_MINOR_VERSION_3 supported) - */ -void mbedtls_ssl_conf_min_version( mbedtls_ssl_config *conf, int major, int minor ); - -#if defined(MBEDTLS_SSL_ENCRYPT_THEN_MAC) -/** - * \brief Enable or disable Encrypt-then-MAC - * (Default: MBEDTLS_SSL_ETM_ENABLED) - * - * \note This should always be enabled, it is a security - * improvement, and should not cause any interoperability - * issue (used only if the peer supports it too). - * - * \param conf SSL configuration - * \param etm MBEDTLS_SSL_ETM_ENABLED or MBEDTLS_SSL_ETM_DISABLED - */ -void mbedtls_ssl_conf_encrypt_then_mac( mbedtls_ssl_config *conf, char etm ); -#endif /* MBEDTLS_SSL_ENCRYPT_THEN_MAC */ - -#if defined(MBEDTLS_SSL_EXTENDED_MASTER_SECRET) -/** - * \brief Enable or disable Extended Master Secret negotiation. - * (Default: MBEDTLS_SSL_EXTENDED_MS_ENABLED) - * - * \note This should always be enabled, it is a security fix to the - * protocol, and should not cause any interoperability issue - * (used only if the peer supports it too). - * - * \param conf SSL configuration - * \param ems MBEDTLS_SSL_EXTENDED_MS_ENABLED or MBEDTLS_SSL_EXTENDED_MS_DISABLED - */ -void mbedtls_ssl_conf_extended_master_secret( mbedtls_ssl_config *conf, char ems ); -#endif /* MBEDTLS_SSL_EXTENDED_MASTER_SECRET */ - -#if defined(MBEDTLS_SSL_SRV_C) -/** - * \brief Whether to send a list of acceptable CAs in - * CertificateRequest messages. - * (Default: do send) - * - * \param conf SSL configuration - * \param cert_req_ca_list MBEDTLS_SSL_CERT_REQ_CA_LIST_ENABLED or - * MBEDTLS_SSL_CERT_REQ_CA_LIST_DISABLED - */ -void mbedtls_ssl_conf_cert_req_ca_list( mbedtls_ssl_config *conf, - char cert_req_ca_list ); -#endif /* MBEDTLS_SSL_SRV_C */ - -#if defined(MBEDTLS_SSL_MAX_FRAGMENT_LENGTH) -/** - * \brief Set the maximum fragment length to emit and/or negotiate. - * (Typical: the smaller of #MBEDTLS_SSL_IN_CONTENT_LEN and - * #MBEDTLS_SSL_OUT_CONTENT_LEN, usually `2^14` bytes) - * (Server: set maximum fragment length to emit, - * usually negotiated by the client during handshake) - * (Client: set maximum fragment length to emit *and* - * negotiate with the server during handshake) - * (Default: #MBEDTLS_SSL_MAX_FRAG_LEN_NONE) - * - * \note On the client side, the maximum fragment length extension - * *will not* be used, unless the maximum fragment length has - * been set via this function to a value different than - * #MBEDTLS_SSL_MAX_FRAG_LEN_NONE. - * - * \note With TLS, this currently only affects ApplicationData (sent - * with \c mbedtls_ssl_read()), not handshake messages. - * With DTLS, this affects both ApplicationData and handshake. - * - * \note This sets the maximum length for a record's payload, - * excluding record overhead that will be added to it, see - * \c mbedtls_ssl_get_record_expansion(). - * - * \note For DTLS, it is also possible to set a limit for the total - * size of daragrams passed to the transport layer, including - * record overhead, see \c mbedtls_ssl_set_mtu(). - * - * \param conf SSL configuration - * \param mfl_code Code for maximum fragment length (allowed values: - * MBEDTLS_SSL_MAX_FRAG_LEN_512, MBEDTLS_SSL_MAX_FRAG_LEN_1024, - * MBEDTLS_SSL_MAX_FRAG_LEN_2048, MBEDTLS_SSL_MAX_FRAG_LEN_4096) - * - * \return 0 if successful or MBEDTLS_ERR_SSL_BAD_INPUT_DATA - */ -int mbedtls_ssl_conf_max_frag_len( mbedtls_ssl_config *conf, unsigned char mfl_code ); -#endif /* MBEDTLS_SSL_MAX_FRAGMENT_LENGTH */ - -#if defined(MBEDTLS_SSL_SRV_C) -/** - * \brief Pick the ciphersuites order according to the second parameter - * in the SSL Server module (MBEDTLS_SSL_SRV_C). - * (Default, if never called: MBEDTLS_SSL_SRV_CIPHERSUITE_ORDER_SERVER) - * - * \param conf SSL configuration - * \param order Server or client (MBEDTLS_SSL_SRV_CIPHERSUITE_ORDER_SERVER - * or MBEDTLS_SSL_SRV_CIPHERSUITE_ORDER_CLIENT) - */ -void mbedtls_ssl_conf_preference_order( mbedtls_ssl_config *conf, int order ); -#endif /* MBEDTLS_SSL_SRV_C */ - -#if defined(MBEDTLS_SSL_SESSION_TICKETS) && defined(MBEDTLS_SSL_CLI_C) -/** - * \brief Enable / Disable session tickets (client only). - * (Default: MBEDTLS_SSL_SESSION_TICKETS_ENABLED.) - * - * \note On server, use \c mbedtls_ssl_conf_session_tickets_cb(). - * - * \param conf SSL configuration - * \param use_tickets Enable or disable (MBEDTLS_SSL_SESSION_TICKETS_ENABLED or - * MBEDTLS_SSL_SESSION_TICKETS_DISABLED) - */ -void mbedtls_ssl_conf_session_tickets( mbedtls_ssl_config *conf, int use_tickets ); -#endif /* MBEDTLS_SSL_SESSION_TICKETS && MBEDTLS_SSL_CLI_C */ - -#if defined(MBEDTLS_SSL_RENEGOTIATION) -/** - * \brief Enable / Disable renegotiation support for connection when - * initiated by peer - * (Default: MBEDTLS_SSL_RENEGOTIATION_DISABLED) - * - * \warning It is recommended to always disable renegotation unless you - * know you need it and you know what you're doing. In the - * past, there have been several issues associated with - * renegotiation or a poor understanding of its properties. - * - * \note Server-side, enabling renegotiation also makes the server - * susceptible to a resource DoS by a malicious client. - * - * \param conf SSL configuration - * \param renegotiation Enable or disable (MBEDTLS_SSL_RENEGOTIATION_ENABLED or - * MBEDTLS_SSL_RENEGOTIATION_DISABLED) - */ -void mbedtls_ssl_conf_renegotiation( mbedtls_ssl_config *conf, int renegotiation ); -#endif /* MBEDTLS_SSL_RENEGOTIATION */ - -/** - * \brief Prevent or allow legacy renegotiation. - * (Default: MBEDTLS_SSL_LEGACY_NO_RENEGOTIATION) - * - * MBEDTLS_SSL_LEGACY_NO_RENEGOTIATION allows connections to - * be established even if the peer does not support - * secure renegotiation, but does not allow renegotiation - * to take place if not secure. - * (Interoperable and secure option) - * - * MBEDTLS_SSL_LEGACY_ALLOW_RENEGOTIATION allows renegotiations - * with non-upgraded peers. Allowing legacy renegotiation - * makes the connection vulnerable to specific man in the - * middle attacks. (See RFC 5746) - * (Most interoperable and least secure option) - * - * MBEDTLS_SSL_LEGACY_BREAK_HANDSHAKE breaks off connections - * if peer does not support secure renegotiation. Results - * in interoperability issues with non-upgraded peers - * that do not support renegotiation altogether. - * (Most secure option, interoperability issues) - * - * \param conf SSL configuration - * \param allow_legacy Prevent or allow (SSL_NO_LEGACY_RENEGOTIATION, - * SSL_ALLOW_LEGACY_RENEGOTIATION or - * MBEDTLS_SSL_LEGACY_BREAK_HANDSHAKE) - */ -void mbedtls_ssl_conf_legacy_renegotiation( mbedtls_ssl_config *conf, int allow_legacy ); - -#if defined(MBEDTLS_SSL_RENEGOTIATION) -/** - * \brief Enforce renegotiation requests. - * (Default: enforced, max_records = 16) - * - * When we request a renegotiation, the peer can comply or - * ignore the request. This function allows us to decide - * whether to enforce our renegotiation requests by closing - * the connection if the peer doesn't comply. - * - * However, records could already be in transit from the peer - * when the request is emitted. In order to increase - * reliability, we can accept a number of records before the - * expected handshake records. - * - * The optimal value is highly dependent on the specific usage - * scenario. - * - * \note With DTLS and server-initiated renegotiation, the - * HelloRequest is retransmited every time mbedtls_ssl_read() times - * out or receives Application Data, until: - * - max_records records have beens seen, if it is >= 0, or - * - the number of retransmits that would happen during an - * actual handshake has been reached. - * Please remember the request might be lost a few times - * if you consider setting max_records to a really low value. - * - * \warning On client, the grace period can only happen during - * mbedtls_ssl_read(), as opposed to mbedtls_ssl_write() and mbedtls_ssl_renegotiate() - * which always behave as if max_record was 0. The reason is, - * if we receive application data from the server, we need a - * place to write it, which only happens during mbedtls_ssl_read(). - * - * \param conf SSL configuration - * \param max_records Use MBEDTLS_SSL_RENEGOTIATION_NOT_ENFORCED if you don't want to - * enforce renegotiation, or a non-negative value to enforce - * it but allow for a grace period of max_records records. - */ -void mbedtls_ssl_conf_renegotiation_enforced( mbedtls_ssl_config *conf, int max_records ); - -/** - * \brief Set record counter threshold for periodic renegotiation. - * (Default: 2^48 - 1) - * - * Renegotiation is automatically triggered when a record - * counter (outgoing or incoming) crosses the defined - * threshold. The default value is meant to prevent the - * connection from being closed when the counter is about to - * reached its maximal value (it is not allowed to wrap). - * - * Lower values can be used to enforce policies such as "keys - * must be refreshed every N packets with cipher X". - * - * The renegotiation period can be disabled by setting - * conf->disable_renegotiation to - * MBEDTLS_SSL_RENEGOTIATION_DISABLED. - * - * \note When the configured transport is - * MBEDTLS_SSL_TRANSPORT_DATAGRAM the maximum renegotiation - * period is 2^48 - 1, and for MBEDTLS_SSL_TRANSPORT_STREAM, - * the maximum renegotiation period is 2^64 - 1. - * - * \param conf SSL configuration - * \param period The threshold value: a big-endian 64-bit number. - */ -void mbedtls_ssl_conf_renegotiation_period( mbedtls_ssl_config *conf, - const unsigned char period[8] ); -#endif /* MBEDTLS_SSL_RENEGOTIATION */ - -/** - * \brief Check if there is data already read from the - * underlying transport but not yet processed. - * - * \param ssl SSL context - * - * \return 0 if nothing's pending, 1 otherwise. - * - * \note This is different in purpose and behaviour from - * \c mbedtls_ssl_get_bytes_avail in that it considers - * any kind of unprocessed data, not only unread - * application data. If \c mbedtls_ssl_get_bytes - * returns a non-zero value, this function will - * also signal pending data, but the converse does - * not hold. For example, in DTLS there might be - * further records waiting to be processed from - * the current underlying transport's datagram. - * - * \note If this function returns 1 (data pending), this - * does not imply that a subsequent call to - * \c mbedtls_ssl_read will provide any data; - * e.g., the unprocessed data might turn out - * to be an alert or a handshake message. - * - * \note This function is useful in the following situation: - * If the SSL/TLS module successfully returns from an - * operation - e.g. a handshake or an application record - * read - and you're awaiting incoming data next, you - * must not immediately idle on the underlying transport - * to have data ready, but you need to check the value - * of this function first. The reason is that the desired - * data might already be read but not yet processed. - * If, in contrast, a previous call to the SSL/TLS module - * returned MBEDTLS_ERR_SSL_WANT_READ, it is not necessary - * to call this function, as the latter error code entails - * that all internal data has been processed. - * - */ -int mbedtls_ssl_check_pending( const mbedtls_ssl_context *ssl ); - -/** - * \brief Return the number of application data bytes - * remaining to be read from the current record. - * - * \param ssl SSL context - * - * \return How many bytes are available in the application - * data record read buffer. - * - * \note When working over a datagram transport, this is - * useful to detect the current datagram's boundary - * in case \c mbedtls_ssl_read has written the maximal - * amount of data fitting into the input buffer. - * - */ -size_t mbedtls_ssl_get_bytes_avail( const mbedtls_ssl_context *ssl ); - -/** - * \brief Return the result of the certificate verification - * - * \param ssl The SSL context to use. - * - * \return \c 0 if the certificate verification was successful. - * \return \c -1u if the result is not available. This may happen - * e.g. if the handshake aborts early, or a verification - * callback returned a fatal error. - * \return A bitwise combination of \c MBEDTLS_X509_BADCERT_XXX - * and \c MBEDTLS_X509_BADCRL_XXX failure flags; see x509.h. - */ -uint32_t mbedtls_ssl_get_verify_result( const mbedtls_ssl_context *ssl ); - -/** - * \brief Return the name of the current ciphersuite - * - * \param ssl SSL context - * - * \return a string containing the ciphersuite name - */ -const char *mbedtls_ssl_get_ciphersuite( const mbedtls_ssl_context *ssl ); - -/** - * \brief Return the current TLS version - * - * \param ssl SSL context - * - * \return a string containing the TLS version - */ -const char *mbedtls_ssl_get_version( const mbedtls_ssl_context *ssl ); - -/** - * \brief Return the (maximum) number of bytes added by the record - * layer: header + encryption/MAC overhead (inc. padding) - * - * \param ssl SSL context - * - * \return Current maximum record expansion in bytes - */ -int mbedtls_ssl_get_record_expansion( const mbedtls_ssl_context *ssl ); - -/** - * \brief Return the current maximum outgoing record payload in bytes. - * - * \note The logic to determine the maximum outgoing record payload is - * version-specific. It takes into account various factors, such as - * the mbedtls_config.h setting \c MBEDTLS_SSL_OUT_CONTENT_LEN, extensions - * such as the max fragment length or record size limit extension if - * used, and for DTLS the path MTU as configured and current - * record expansion. - * - * \note With DTLS, \c mbedtls_ssl_write() will return an error if - * called with a larger length value. - * With TLS, \c mbedtls_ssl_write() will fragment the input if - * necessary and return the number of bytes written; it is up - * to the caller to call \c mbedtls_ssl_write() again in - * order to send the remaining bytes if any. - * - * \sa mbedtls_ssl_get_max_out_record_payload() - * \sa mbedtls_ssl_get_record_expansion() - * - * \param ssl SSL context - * - * \return Current maximum payload for an outgoing record, - * or a negative error code. - */ -int mbedtls_ssl_get_max_out_record_payload( const mbedtls_ssl_context *ssl ); - -/** - * \brief Return the current maximum incoming record payload in bytes. - * - * \note The logic to determine the maximum outgoing record payload is - * version-specific. It takes into account various factors, such as - * the mbedtls_config.h setting \c MBEDTLS_SSL_IN_CONTENT_LEN, extensions - * such as the max fragment length extension or record size limit - * extension if used, and the current record expansion. - * - * \sa mbedtls_ssl_set_mtu() - * \sa mbedtls_ssl_get_max_in_record_payload() - * \sa mbedtls_ssl_get_record_expansion() - * - * \param ssl SSL context - * - * \return Current maximum payload for an outgoing record, - * or a negative error code. - */ -int mbedtls_ssl_get_max_in_record_payload( const mbedtls_ssl_context *ssl ); - -#if defined(MBEDTLS_X509_CRT_PARSE_C) -/** - * \brief Return the peer certificate from the current connection. - * - * \param ssl The SSL context to use. This must be initialized and setup. - * - * \return The current peer certificate, if available. - * The returned certificate is owned by the SSL context and - * is valid only until the next call to the SSL API. - * \return \c NULL if no peer certificate is available. This might - * be because the chosen ciphersuite doesn't use CRTs - * (PSK-based ciphersuites, for example), or because - * #MBEDTLS_SSL_KEEP_PEER_CERTIFICATE has been disabled, - * allowing the stack to free the peer's CRT to save memory. - * - * \note For one-time inspection of the peer's certificate during - * the handshake, consider registering an X.509 CRT verification - * callback through mbedtls_ssl_conf_verify() instead of calling - * this function. Using mbedtls_ssl_conf_verify() also comes at - * the benefit of allowing you to influence the verification - * process, for example by masking expected and tolerated - * verification failures. - * - * \warning You must not use the pointer returned by this function - * after any further call to the SSL API, including - * mbedtls_ssl_read() and mbedtls_ssl_write(); this is - * because the pointer might change during renegotiation, - * which happens transparently to the user. - * If you want to use the certificate across API calls, - * you must make a copy. - */ -const mbedtls_x509_crt *mbedtls_ssl_get_peer_cert( const mbedtls_ssl_context *ssl ); -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - -#if defined(MBEDTLS_SSL_CLI_C) -/** - * \brief Export a session in order to resume it later. - * - * \param ssl The SSL context representing the connection for which to - * to export a session structure for later resumption. - * \param session The target structure in which to store the exported session. - * This must have been initialized with mbedtls_ssl_init_session() - * but otherwise be unused. - * - * \note This function can handle a variety of mechanisms for session - * resumption: For TLS 1.2, both session ID-based resumption and - * ticket-based resumption will be considered. For TLS 1.3, - * once implemented, sessions equate to tickets, and calling - * this function multiple times will export the available - * tickets one a time until no further tickets are available, - * in which case MBEDTLS_ERR_SSL_FEATURE_UNAVAILABLE will - * be returned. - * - * \note Calling this function multiple times will only be useful - * once TLS 1.3 is supported. For TLS 1.2 connections, this - * function should be called at most once. - * - * \return \c 0 if successful. In this case, \p session can be used for - * session resumption by passing it to mbedtls_ssl_set_session(), - * and serialized for storage via mbedtls_ssl_session_save(). - * \return #MBEDTLS_ERR_SSL_FEATURE_UNAVAILABLE if no further session - * is available for export. - * This error is a non-fatal, and has no observable effect on - * the SSL context or the destination session. - * \return Another negative error code on other kinds of failure. - * - * \sa mbedtls_ssl_set_session() - * \sa mbedtls_ssl_session_save() - */ -int mbedtls_ssl_get_session( const mbedtls_ssl_context *ssl, - mbedtls_ssl_session *session ); -#endif /* MBEDTLS_SSL_CLI_C */ - -/** - * \brief Perform the SSL handshake - * - * \param ssl SSL context - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_SSL_WANT_READ or #MBEDTLS_ERR_SSL_WANT_WRITE - * if the handshake is incomplete and waiting for data to - * be available for reading from or writing to the underlying - * transport - in this case you must call this function again - * when the underlying transport is ready for the operation. - * \return #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS if an asynchronous - * operation is in progress (see - * mbedtls_ssl_conf_async_private_cb()) - in this case you - * must call this function again when the operation is ready. - * \return #MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS if a cryptographic - * operation is in progress (see mbedtls_ecp_set_max_ops()) - - * in this case you must call this function again to complete - * the handshake when you're done attending other tasks. - * \return #MBEDTLS_ERR_SSL_HELLO_VERIFY_REQUIRED if DTLS is in use - * and the client did not demonstrate reachability yet - in - * this case you must stop using the context (see below). - * \return Another SSL error code - in this case you must stop using - * the context (see below). - * - * \warning If this function returns something other than - * \c 0, - * #MBEDTLS_ERR_SSL_WANT_READ, - * #MBEDTLS_ERR_SSL_WANT_WRITE, - * #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS or - * #MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS, - * you must stop using the SSL context for reading or writing, - * and either free it or call \c mbedtls_ssl_session_reset() - * on it before re-using it for a new connection; the current - * connection must be closed. - * - * \note If DTLS is in use, then you may choose to handle - * #MBEDTLS_ERR_SSL_HELLO_VERIFY_REQUIRED specially for logging - * purposes, as it is an expected return value rather than an - * actual error, but you still need to reset/free the context. - * - * \note Remarks regarding event-driven DTLS: - * If the function returns #MBEDTLS_ERR_SSL_WANT_READ, no datagram - * from the underlying transport layer is currently being processed, - * and it is safe to idle until the timer or the underlying transport - * signal a new event. This is not true for a successful handshake, - * in which case the datagram of the underlying transport that is - * currently being processed might or might not contain further - * DTLS records. - */ -int mbedtls_ssl_handshake( mbedtls_ssl_context *ssl ); - -/** - * \brief Perform a single step of the SSL handshake - * - * \note The state of the context (ssl->state) will be at - * the next state after this function returns \c 0. Do not - * call this function if state is MBEDTLS_SSL_HANDSHAKE_OVER. - * - * \param ssl SSL context - * - * \return See mbedtls_ssl_handshake(). - * - * \warning If this function returns something other than \c 0, - * #MBEDTLS_ERR_SSL_WANT_READ, #MBEDTLS_ERR_SSL_WANT_WRITE, - * #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS or - * #MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS, you must stop using - * the SSL context for reading or writing, and either free it - * or call \c mbedtls_ssl_session_reset() on it before - * re-using it for a new connection; the current connection - * must be closed. - */ -int mbedtls_ssl_handshake_step( mbedtls_ssl_context *ssl ); - -#if defined(MBEDTLS_SSL_RENEGOTIATION) -/** - * \brief Initiate an SSL renegotiation on the running connection. - * Client: perform the renegotiation right now. - * Server: request renegotiation, which will be performed - * during the next call to mbedtls_ssl_read() if honored by - * client. - * - * \param ssl SSL context - * - * \return 0 if successful, or any mbedtls_ssl_handshake() return - * value except #MBEDTLS_ERR_SSL_CLIENT_RECONNECT that can't - * happen during a renegotiation. - * - * \warning If this function returns something other than \c 0, - * #MBEDTLS_ERR_SSL_WANT_READ, #MBEDTLS_ERR_SSL_WANT_WRITE, - * #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS or - * #MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS, you must stop using - * the SSL context for reading or writing, and either free it - * or call \c mbedtls_ssl_session_reset() on it before - * re-using it for a new connection; the current connection - * must be closed. - * - */ -int mbedtls_ssl_renegotiate( mbedtls_ssl_context *ssl ); -#endif /* MBEDTLS_SSL_RENEGOTIATION */ - -/** - * \brief Read at most 'len' application data bytes - * - * \param ssl SSL context - * \param buf buffer that will hold the data - * \param len maximum number of bytes to read - * - * \return The (positive) number of bytes read if successful. - * \return \c 0 if the read end of the underlying transport was closed - * without sending a CloseNotify beforehand, which might happen - * because of various reasons (internal error of an underlying - * stack, non-conformant peer not sending a CloseNotify and - * such) - in this case you must stop using the context - * (see below). - * \return #MBEDTLS_ERR_SSL_PEER_CLOSE_NOTIFY if the underlying - * transport is still functional, but the peer has - * acknowledged to not send anything anymore. - * \return #MBEDTLS_ERR_SSL_WANT_READ or #MBEDTLS_ERR_SSL_WANT_WRITE - * if the handshake is incomplete and waiting for data to - * be available for reading from or writing to the underlying - * transport - in this case you must call this function again - * when the underlying transport is ready for the operation. - * \return #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS if an asynchronous - * operation is in progress (see - * mbedtls_ssl_conf_async_private_cb()) - in this case you - * must call this function again when the operation is ready. - * \return #MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS if a cryptographic - * operation is in progress (see mbedtls_ecp_set_max_ops()) - - * in this case you must call this function again to complete - * the handshake when you're done attending other tasks. - * \return #MBEDTLS_ERR_SSL_CLIENT_RECONNECT if we're at the server - * side of a DTLS connection and the client is initiating a - * new connection using the same source port. See below. - * \return Another SSL error code - in this case you must stop using - * the context (see below). - * - * \warning If this function returns something other than - * a positive value, - * #MBEDTLS_ERR_SSL_WANT_READ, - * #MBEDTLS_ERR_SSL_WANT_WRITE, - * #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS, - * #MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS or - * #MBEDTLS_ERR_SSL_CLIENT_RECONNECT, - * you must stop using the SSL context for reading or writing, - * and either free it or call \c mbedtls_ssl_session_reset() - * on it before re-using it for a new connection; the current - * connection must be closed. - * - * \note When this function returns #MBEDTLS_ERR_SSL_CLIENT_RECONNECT - * (which can only happen server-side), it means that a client - * is initiating a new connection using the same source port. - * You can either treat that as a connection close and wait - * for the client to resend a ClientHello, or directly - * continue with \c mbedtls_ssl_handshake() with the same - * context (as it has been reset internally). Either way, you - * must make sure this is seen by the application as a new - * connection: application state, if any, should be reset, and - * most importantly the identity of the client must be checked - * again. WARNING: not validating the identity of the client - * again, or not transmitting the new identity to the - * application layer, would allow authentication bypass! - * - * \note Remarks regarding event-driven DTLS: - * - If the function returns #MBEDTLS_ERR_SSL_WANT_READ, no datagram - * from the underlying transport layer is currently being processed, - * and it is safe to idle until the timer or the underlying transport - * signal a new event. - * - This function may return MBEDTLS_ERR_SSL_WANT_READ even if data was - * initially available on the underlying transport, as this data may have - * been only e.g. duplicated messages or a renegotiation request. - * Therefore, you must be prepared to receive MBEDTLS_ERR_SSL_WANT_READ even - * when reacting to an incoming-data event from the underlying transport. - * - On success, the datagram of the underlying transport that is currently - * being processed may contain further DTLS records. You should call - * \c mbedtls_ssl_check_pending to check for remaining records. - * - */ -int mbedtls_ssl_read( mbedtls_ssl_context *ssl, unsigned char *buf, size_t len ); - -/** - * \brief Try to write exactly 'len' application data bytes - * - * \warning This function will do partial writes in some cases. If the - * return value is non-negative but less than length, the - * function must be called again with updated arguments: - * buf + ret, len - ret (if ret is the return value) until - * it returns a value equal to the last 'len' argument. - * - * \param ssl SSL context - * \param buf buffer holding the data - * \param len how many bytes must be written - * - * \return The (non-negative) number of bytes actually written if - * successful (may be less than \p len). - * \return #MBEDTLS_ERR_SSL_WANT_READ or #MBEDTLS_ERR_SSL_WANT_WRITE - * if the handshake is incomplete and waiting for data to - * be available for reading from or writing to the underlying - * transport - in this case you must call this function again - * when the underlying transport is ready for the operation. - * \return #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS if an asynchronous - * operation is in progress (see - * mbedtls_ssl_conf_async_private_cb()) - in this case you - * must call this function again when the operation is ready. - * \return #MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS if a cryptographic - * operation is in progress (see mbedtls_ecp_set_max_ops()) - - * in this case you must call this function again to complete - * the handshake when you're done attending other tasks. - * \return Another SSL error code - in this case you must stop using - * the context (see below). - * - * \warning If this function returns something other than - * a non-negative value, - * #MBEDTLS_ERR_SSL_WANT_READ, - * #MBEDTLS_ERR_SSL_WANT_WRITE, - * #MBEDTLS_ERR_SSL_ASYNC_IN_PROGRESS or - * #MBEDTLS_ERR_SSL_CRYPTO_IN_PROGRESS, - * you must stop using the SSL context for reading or writing, - * and either free it or call \c mbedtls_ssl_session_reset() - * on it before re-using it for a new connection; the current - * connection must be closed. - * - * \note When this function returns #MBEDTLS_ERR_SSL_WANT_WRITE/READ, - * it must be called later with the *same* arguments, - * until it returns a value greater that or equal to 0. When - * the function returns #MBEDTLS_ERR_SSL_WANT_WRITE there may be - * some partial data in the output buffer, however this is not - * yet sent. - * - * \note If the requested length is greater than the maximum - * fragment length (either the built-in limit or the one set - * or negotiated with the peer), then: - * - with TLS, less bytes than requested are written. - * - with DTLS, MBEDTLS_ERR_SSL_BAD_INPUT_DATA is returned. - * \c mbedtls_ssl_get_max_out_record_payload() may be used to - * query the active maximum fragment length. - * - * \note Attempting to write 0 bytes will result in an empty TLS - * application record being sent. - */ -int mbedtls_ssl_write( mbedtls_ssl_context *ssl, const unsigned char *buf, size_t len ); - -/** - * \brief Send an alert message - * - * \param ssl SSL context - * \param level The alert level of the message - * (MBEDTLS_SSL_ALERT_LEVEL_WARNING or MBEDTLS_SSL_ALERT_LEVEL_FATAL) - * \param message The alert message (SSL_ALERT_MSG_*) - * - * \return 0 if successful, or a specific SSL error code. - * - * \note If this function returns something other than 0 or - * MBEDTLS_ERR_SSL_WANT_READ/WRITE, you must stop using - * the SSL context for reading or writing, and either free it or - * call \c mbedtls_ssl_session_reset() on it before re-using it - * for a new connection; the current connection must be closed. - */ -int mbedtls_ssl_send_alert_message( mbedtls_ssl_context *ssl, - unsigned char level, - unsigned char message ); -/** - * \brief Notify the peer that the connection is being closed - * - * \param ssl SSL context - * - * \return 0 if successful, or a specific SSL error code. - * - * \note If this function returns something other than 0 or - * MBEDTLS_ERR_SSL_WANT_READ/WRITE, you must stop using - * the SSL context for reading or writing, and either free it or - * call \c mbedtls_ssl_session_reset() on it before re-using it - * for a new connection; the current connection must be closed. - */ -int mbedtls_ssl_close_notify( mbedtls_ssl_context *ssl ); - -/** - * \brief Free referenced items in an SSL context and clear memory - * - * \param ssl SSL context - */ -void mbedtls_ssl_free( mbedtls_ssl_context *ssl ); - -#if defined(MBEDTLS_SSL_CONTEXT_SERIALIZATION) -/** - * \brief Save an active connection as serialized data in a buffer. - * This allows the freeing or re-using of the SSL context - * while still picking up the connection later in a way that - * it entirely transparent to the peer. - * - * \see mbedtls_ssl_context_load() - * - * \note This feature is currently only available under certain - * conditions, see the documentation of the return value - * #MBEDTLS_ERR_SSL_BAD_INPUT_DATA for details. - * - * \note When this function succeeds, it calls - * mbedtls_ssl_session_reset() on \p ssl which as a result is - * no longer associated with the connection that has been - * serialized. This avoids creating copies of the connection - * state. You're then free to either re-use the context - * structure for a different connection, or call - * mbedtls_ssl_free() on it. See the documentation of - * mbedtls_ssl_session_reset() for more details. - * - * \param ssl The SSL context to save. On success, it is no longer - * associated with the connection that has been serialized. - * \param buf The buffer to write the serialized data to. It must be a - * writeable buffer of at least \p buf_len bytes, or may be \c - * NULL if \p buf_len is \c 0. - * \param buf_len The number of bytes available for writing in \p buf. - * \param olen The size in bytes of the data that has been or would have - * been written. It must point to a valid \c size_t. - * - * \note \p olen is updated to the correct value regardless of - * whether \p buf_len was large enough. This makes it possible - * to determine the necessary size by calling this function - * with \p buf set to \c NULL and \p buf_len to \c 0. However, - * the value of \p olen is only guaranteed to be correct when - * the function returns #MBEDTLS_ERR_SSL_BUFFER_TOO_SMALL or - * \c 0. If the return value is different, then the value of - * \p olen is undefined. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_SSL_BUFFER_TOO_SMALL if \p buf is too small. - * \return #MBEDTLS_ERR_SSL_ALLOC_FAILED if memory allocation failed - * while reseting the context. - * \return #MBEDTLS_ERR_SSL_BAD_INPUT_DATA if a handshake is in - * progress, or there is pending data for reading or sending, - * or the connection does not use DTLS 1.2 with an AEAD - * ciphersuite, or renegotiation is enabled. - */ -int mbedtls_ssl_context_save( mbedtls_ssl_context *ssl, - unsigned char *buf, - size_t buf_len, - size_t *olen ); - -/** - * \brief Load serialized connection data to an SSL context. - * - * \see mbedtls_ssl_context_save() - * - * \warning The same serialized data must never be loaded into more - * that one context. In order to ensure that, after - * successfully loading serialized data to an SSL context, you - * should immediately destroy or invalidate all copies of the - * serialized data that was loaded. Loading the same data in - * more than one context would cause severe security failures - * including but not limited to loss of confidentiality. - * - * \note Before calling this function, the SSL context must be - * prepared in one of the two following ways. The first way is - * to take a context freshly initialised with - * mbedtls_ssl_init() and call mbedtls_ssl_setup() on it with - * the same ::mbedtls_ssl_config structure that was used in - * the original connection. The second way is to - * call mbedtls_ssl_session_reset() on a context that was - * previously prepared as above but used in the meantime. - * Either way, you must not use the context to perform a - * handshake between calling mbedtls_ssl_setup() or - * mbedtls_ssl_session_reset() and calling this function. You - * may however call other setter functions in that time frame - * as indicated in the note below. - * - * \note Before or after calling this function successfully, you - * also need to configure some connection-specific callbacks - * and settings before you can use the connection again - * (unless they were already set before calling - * mbedtls_ssl_session_reset() and the values are suitable for - * the present connection). Specifically, you want to call - * at least mbedtls_ssl_set_bio() and - * mbedtls_ssl_set_timer_cb(). All other SSL setter functions - * are not necessary to call, either because they're only used - * in handshakes, or because the setting is already saved. You - * might choose to call them anyway, for example in order to - * share code between the cases of establishing a new - * connection and the case of loading an already-established - * connection. - * - * \note If you have new information about the path MTU, you want to - * call mbedtls_ssl_set_mtu() after calling this function, as - * otherwise this function would overwrite your - * newly-configured value with the value that was active when - * the context was saved. - * - * \note When this function returns an error code, it calls - * mbedtls_ssl_free() on \p ssl. In this case, you need to - * prepare the context with the usual sequence starting with a - * call to mbedtls_ssl_init() if you want to use it again. - * - * \param ssl The SSL context structure to be populated. It must have - * been prepared as described in the note above. - * \param buf The buffer holding the serialized connection data. It must - * be a readable buffer of at least \p len bytes. - * \param len The size of the serialized data in bytes. - * - * \return \c 0 if successful. - * \return #MBEDTLS_ERR_SSL_ALLOC_FAILED if memory allocation failed. - * \return #MBEDTLS_ERR_SSL_VERSION_MISMATCH if the serialized data - * comes from a different Mbed TLS version or build. - * \return #MBEDTLS_ERR_SSL_BAD_INPUT_DATA if input data is invalid. - */ -int mbedtls_ssl_context_load( mbedtls_ssl_context *ssl, - const unsigned char *buf, - size_t len ); -#endif /* MBEDTLS_SSL_CONTEXT_SERIALIZATION */ - -/** - * \brief Initialize an SSL configuration context - * Just makes the context ready for - * mbedtls_ssl_config_defaults() or mbedtls_ssl_config_free(). - * - * \note You need to call mbedtls_ssl_config_defaults() unless you - * manually set all of the relevant fields yourself. - * - * \param conf SSL configuration context - */ -void mbedtls_ssl_config_init( mbedtls_ssl_config *conf ); - -/** - * \brief Load reasonnable default SSL configuration values. - * (You need to call mbedtls_ssl_config_init() first.) - * - * \param conf SSL configuration context - * \param endpoint MBEDTLS_SSL_IS_CLIENT or MBEDTLS_SSL_IS_SERVER - * \param transport MBEDTLS_SSL_TRANSPORT_STREAM for TLS, or - * MBEDTLS_SSL_TRANSPORT_DATAGRAM for DTLS - * \param preset a MBEDTLS_SSL_PRESET_XXX value - * - * \note See \c mbedtls_ssl_conf_transport() for notes on DTLS. - * - * \return 0 if successful, or - * MBEDTLS_ERR_XXX_ALLOC_FAILED on memory allocation error. - */ -int mbedtls_ssl_config_defaults( mbedtls_ssl_config *conf, - int endpoint, int transport, int preset ); - -/** - * \brief Free an SSL configuration context - * - * \param conf SSL configuration context - */ -void mbedtls_ssl_config_free( mbedtls_ssl_config *conf ); - -/** - * \brief Initialize SSL session structure - * - * \param session SSL session - */ -void mbedtls_ssl_session_init( mbedtls_ssl_session *session ); - -/** - * \brief Free referenced items in an SSL session including the - * peer certificate and clear memory - * - * \note A session object can be freed even if the SSL context - * that was used to retrieve the session is still in use. - * - * \param session SSL session - */ -void mbedtls_ssl_session_free( mbedtls_ssl_session *session ); - -/** - * \brief TLS-PRF function for key derivation. - * - * \param prf The tls_prf type function type to be used. - * \param secret Secret for the key derivation function. - * \param slen Length of the secret. - * \param label String label for the key derivation function, - * terminated with null character. - * \param random Random bytes. - * \param rlen Length of the random bytes buffer. - * \param dstbuf The buffer holding the derived key. - * \param dlen Length of the output buffer. - * - * \return 0 on success. An SSL specific error on failure. - */ -int mbedtls_ssl_tls_prf( const mbedtls_tls_prf_types prf, - const unsigned char *secret, size_t slen, - const char *label, - const unsigned char *random, size_t rlen, - unsigned char *dstbuf, size_t dlen ); - -#ifdef __cplusplus -} -#endif - -#endif /* ssl.h */ diff --git a/windows/mbedtls/ssl_cache.h b/windows/mbedtls/ssl_cache.h deleted file mode 100644 index 6a81ac99..00000000 --- a/windows/mbedtls/ssl_cache.h +++ /dev/null @@ -1,161 +0,0 @@ -/** - * \file ssl_cache.h - * - * \brief SSL session cache implementation - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_SSL_CACHE_H -#define MBEDTLS_SSL_CACHE_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/ssl.h" - -#if defined(MBEDTLS_THREADING_C) -#include "mbedtls/threading.h" -#endif - -/** - * \name SECTION: Module settings - * - * The configuration options you can set for this module are in this section. - * Either change them in mbedtls_config.h or define them on the compiler command line. - * \{ - */ - -#if !defined(MBEDTLS_SSL_CACHE_DEFAULT_TIMEOUT) -#define MBEDTLS_SSL_CACHE_DEFAULT_TIMEOUT 86400 /*!< 1 day */ -#endif - -#if !defined(MBEDTLS_SSL_CACHE_DEFAULT_MAX_ENTRIES) -#define MBEDTLS_SSL_CACHE_DEFAULT_MAX_ENTRIES 50 /*!< Maximum entries in cache */ -#endif - -/* \} name SECTION: Module settings */ - -#ifdef __cplusplus -extern "C" { -#endif - -typedef struct mbedtls_ssl_cache_context mbedtls_ssl_cache_context; -typedef struct mbedtls_ssl_cache_entry mbedtls_ssl_cache_entry; - -/** - * \brief This structure is used for storing cache entries - */ -struct mbedtls_ssl_cache_entry -{ -#if defined(MBEDTLS_HAVE_TIME) - mbedtls_time_t MBEDTLS_PRIVATE(timestamp); /*!< entry timestamp */ -#endif - - unsigned char MBEDTLS_PRIVATE(session_id)[32]; /*!< session ID */ - size_t MBEDTLS_PRIVATE(session_id_len); - - unsigned char *MBEDTLS_PRIVATE(session); /*!< serialized session */ - size_t MBEDTLS_PRIVATE(session_len); - - mbedtls_ssl_cache_entry *MBEDTLS_PRIVATE(next); /*!< chain pointer */ -}; - -/** - * \brief Cache context - */ -struct mbedtls_ssl_cache_context -{ - mbedtls_ssl_cache_entry *MBEDTLS_PRIVATE(chain); /*!< start of the chain */ - int MBEDTLS_PRIVATE(timeout); /*!< cache entry timeout */ - int MBEDTLS_PRIVATE(max_entries); /*!< maximum entries */ -#if defined(MBEDTLS_THREADING_C) - mbedtls_threading_mutex_t MBEDTLS_PRIVATE(mutex); /*!< mutex */ -#endif -}; - -/** - * \brief Initialize an SSL cache context - * - * \param cache SSL cache context - */ -void mbedtls_ssl_cache_init( mbedtls_ssl_cache_context *cache ); - -/** - * \brief Cache get callback implementation - * (Thread-safe if MBEDTLS_THREADING_C is enabled) - * - * \param data The SSL cache context to use. - * \param session_id The pointer to the buffer holding the session ID - * for the session to load. - * \param session_id_len The length of \p session_id in bytes. - * \param session The address at which to store the session - * associated with \p session_id, if present. - */ -int mbedtls_ssl_cache_get( void *data, - unsigned char const *session_id, - size_t session_id_len, - mbedtls_ssl_session *session ); - -/** - * \brief Cache set callback implementation - * (Thread-safe if MBEDTLS_THREADING_C is enabled) - * - * \param data The SSL cache context to use. - * \param session_id The pointer to the buffer holding the session ID - * associated to \p session. - * \param session_id_len The length of \p session_id in bytes. - * \param session The session to store. - */ -int mbedtls_ssl_cache_set( void *data, - unsigned char const *session_id, - size_t session_id_len, - const mbedtls_ssl_session *session ); - -#if defined(MBEDTLS_HAVE_TIME) -/** - * \brief Set the cache timeout - * (Default: MBEDTLS_SSL_CACHE_DEFAULT_TIMEOUT (1 day)) - * - * A timeout of 0 indicates no timeout. - * - * \param cache SSL cache context - * \param timeout cache entry timeout in seconds - */ -void mbedtls_ssl_cache_set_timeout( mbedtls_ssl_cache_context *cache, int timeout ); -#endif /* MBEDTLS_HAVE_TIME */ - -/** - * \brief Set the maximum number of cache entries - * (Default: MBEDTLS_SSL_CACHE_DEFAULT_MAX_ENTRIES (50)) - * - * \param cache SSL cache context - * \param max cache entry maximum - */ -void mbedtls_ssl_cache_set_max_entries( mbedtls_ssl_cache_context *cache, int max ); - -/** - * \brief Free referenced items in a cache context and clear memory - * - * \param cache SSL cache context - */ -void mbedtls_ssl_cache_free( mbedtls_ssl_cache_context *cache ); - -#ifdef __cplusplus -} -#endif - -#endif /* ssl_cache.h */ diff --git a/windows/mbedtls/ssl_ciphersuites.h b/windows/mbedtls/ssl_ciphersuites.h deleted file mode 100644 index 812560c8..00000000 --- a/windows/mbedtls/ssl_ciphersuites.h +++ /dev/null @@ -1,528 +0,0 @@ -/** - * \file ssl_ciphersuites.h - * - * \brief SSL Ciphersuites for mbed TLS - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_SSL_CIPHERSUITES_H -#define MBEDTLS_SSL_CIPHERSUITES_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/pk.h" -#include "mbedtls/cipher.h" -#include "mbedtls/md.h" - -#ifdef __cplusplus -extern "C" { -#endif - -/* - * Supported ciphersuites (Official IANA names) - */ -#define MBEDTLS_TLS_RSA_WITH_NULL_MD5 0x01 /**< Weak! */ -#define MBEDTLS_TLS_RSA_WITH_NULL_SHA 0x02 /**< Weak! */ - -#define MBEDTLS_TLS_PSK_WITH_NULL_SHA 0x2C /**< Weak! */ -#define MBEDTLS_TLS_DHE_PSK_WITH_NULL_SHA 0x2D /**< Weak! */ -#define MBEDTLS_TLS_RSA_PSK_WITH_NULL_SHA 0x2E /**< Weak! */ -#define MBEDTLS_TLS_RSA_WITH_AES_128_CBC_SHA 0x2F - -#define MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CBC_SHA 0x33 -#define MBEDTLS_TLS_RSA_WITH_AES_256_CBC_SHA 0x35 -#define MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CBC_SHA 0x39 - -#define MBEDTLS_TLS_RSA_WITH_NULL_SHA256 0x3B /**< Weak! */ -#define MBEDTLS_TLS_RSA_WITH_AES_128_CBC_SHA256 0x3C /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_WITH_AES_256_CBC_SHA256 0x3D /**< TLS 1.2 */ - -#define MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_CBC_SHA 0x41 -#define MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA 0x45 - -#define MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CBC_SHA256 0x67 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CBC_SHA256 0x6B /**< TLS 1.2 */ - -#define MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_CBC_SHA 0x84 -#define MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA 0x88 - -#define MBEDTLS_TLS_PSK_WITH_AES_128_CBC_SHA 0x8C -#define MBEDTLS_TLS_PSK_WITH_AES_256_CBC_SHA 0x8D - -#define MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CBC_SHA 0x90 -#define MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CBC_SHA 0x91 - -#define MBEDTLS_TLS_RSA_PSK_WITH_AES_128_CBC_SHA 0x94 -#define MBEDTLS_TLS_RSA_PSK_WITH_AES_256_CBC_SHA 0x95 - -#define MBEDTLS_TLS_RSA_WITH_AES_128_GCM_SHA256 0x9C /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_WITH_AES_256_GCM_SHA384 0x9D /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_AES_128_GCM_SHA256 0x9E /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_AES_256_GCM_SHA384 0x9F /**< TLS 1.2 */ - -#define MBEDTLS_TLS_PSK_WITH_AES_128_GCM_SHA256 0xA8 /**< TLS 1.2 */ -#define MBEDTLS_TLS_PSK_WITH_AES_256_GCM_SHA384 0xA9 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_AES_128_GCM_SHA256 0xAA /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_AES_256_GCM_SHA384 0xAB /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_PSK_WITH_AES_128_GCM_SHA256 0xAC /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_PSK_WITH_AES_256_GCM_SHA384 0xAD /**< TLS 1.2 */ - -#define MBEDTLS_TLS_PSK_WITH_AES_128_CBC_SHA256 0xAE -#define MBEDTLS_TLS_PSK_WITH_AES_256_CBC_SHA384 0xAF -#define MBEDTLS_TLS_PSK_WITH_NULL_SHA256 0xB0 /**< Weak! */ -#define MBEDTLS_TLS_PSK_WITH_NULL_SHA384 0xB1 /**< Weak! */ - -#define MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CBC_SHA256 0xB2 -#define MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CBC_SHA384 0xB3 -#define MBEDTLS_TLS_DHE_PSK_WITH_NULL_SHA256 0xB4 /**< Weak! */ -#define MBEDTLS_TLS_DHE_PSK_WITH_NULL_SHA384 0xB5 /**< Weak! */ - -#define MBEDTLS_TLS_RSA_PSK_WITH_AES_128_CBC_SHA256 0xB6 -#define MBEDTLS_TLS_RSA_PSK_WITH_AES_256_CBC_SHA384 0xB7 -#define MBEDTLS_TLS_RSA_PSK_WITH_NULL_SHA256 0xB8 /**< Weak! */ -#define MBEDTLS_TLS_RSA_PSK_WITH_NULL_SHA384 0xB9 /**< Weak! */ - -#define MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_CBC_SHA256 0xBA /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_CBC_SHA256 0xBE /**< TLS 1.2 */ - -#define MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_CBC_SHA256 0xC0 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_CBC_SHA256 0xC4 /**< TLS 1.2 */ - -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_NULL_SHA 0xC001 /**< Weak! */ -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA 0xC004 -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA 0xC005 - -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_NULL_SHA 0xC006 /**< Weak! */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA 0xC009 -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA 0xC00A - -#define MBEDTLS_TLS_ECDH_RSA_WITH_NULL_SHA 0xC00B /**< Weak! */ -#define MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_CBC_SHA 0xC00E -#define MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_CBC_SHA 0xC00F - -#define MBEDTLS_TLS_ECDHE_RSA_WITH_NULL_SHA 0xC010 /**< Weak! */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA 0xC013 -#define MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA 0xC014 - -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA256 0xC023 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA384 0xC024 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA256 0xC025 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_CBC_SHA384 0xC026 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA256 0xC027 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA384 0xC028 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_CBC_SHA256 0xC029 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_CBC_SHA384 0xC02A /**< TLS 1.2 */ - -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256 0xC02B /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384 0xC02C /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_128_GCM_SHA256 0xC02D /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_AES_256_GCM_SHA384 0xC02E /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256 0xC02F /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384 0xC030 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_RSA_WITH_AES_128_GCM_SHA256 0xC031 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_RSA_WITH_AES_256_GCM_SHA384 0xC032 /**< TLS 1.2 */ - -#define MBEDTLS_TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA 0xC035 -#define MBEDTLS_TLS_ECDHE_PSK_WITH_AES_256_CBC_SHA 0xC036 -#define MBEDTLS_TLS_ECDHE_PSK_WITH_AES_128_CBC_SHA256 0xC037 -#define MBEDTLS_TLS_ECDHE_PSK_WITH_AES_256_CBC_SHA384 0xC038 -#define MBEDTLS_TLS_ECDHE_PSK_WITH_NULL_SHA 0xC039 -#define MBEDTLS_TLS_ECDHE_PSK_WITH_NULL_SHA256 0xC03A -#define MBEDTLS_TLS_ECDHE_PSK_WITH_NULL_SHA384 0xC03B - -#define MBEDTLS_TLS_RSA_WITH_ARIA_128_CBC_SHA256 0xC03C /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_WITH_ARIA_256_CBC_SHA384 0xC03D /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_ARIA_128_CBC_SHA256 0xC044 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_ARIA_256_CBC_SHA384 0xC045 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_128_CBC_SHA256 0xC048 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_256_CBC_SHA384 0xC049 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_128_CBC_SHA256 0xC04A /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_256_CBC_SHA384 0xC04B /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_128_CBC_SHA256 0xC04C /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_256_CBC_SHA384 0xC04D /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_128_CBC_SHA256 0xC04E /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_256_CBC_SHA384 0xC04F /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_WITH_ARIA_128_GCM_SHA256 0xC050 /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_WITH_ARIA_256_GCM_SHA384 0xC051 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_ARIA_128_GCM_SHA256 0xC052 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_ARIA_256_GCM_SHA384 0xC053 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_128_GCM_SHA256 0xC05C /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_ARIA_256_GCM_SHA384 0xC05D /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_128_GCM_SHA256 0xC05E /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_ARIA_256_GCM_SHA384 0xC05F /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_128_GCM_SHA256 0xC060 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_ARIA_256_GCM_SHA384 0xC061 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_128_GCM_SHA256 0xC062 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_RSA_WITH_ARIA_256_GCM_SHA384 0xC063 /**< TLS 1.2 */ -#define MBEDTLS_TLS_PSK_WITH_ARIA_128_CBC_SHA256 0xC064 /**< TLS 1.2 */ -#define MBEDTLS_TLS_PSK_WITH_ARIA_256_CBC_SHA384 0xC065 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_ARIA_128_CBC_SHA256 0xC066 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_ARIA_256_CBC_SHA384 0xC067 /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_PSK_WITH_ARIA_128_CBC_SHA256 0xC068 /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_PSK_WITH_ARIA_256_CBC_SHA384 0xC069 /**< TLS 1.2 */ -#define MBEDTLS_TLS_PSK_WITH_ARIA_128_GCM_SHA256 0xC06A /**< TLS 1.2 */ -#define MBEDTLS_TLS_PSK_WITH_ARIA_256_GCM_SHA384 0xC06B /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_ARIA_128_GCM_SHA256 0xC06C /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_ARIA_256_GCM_SHA384 0xC06D /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_PSK_WITH_ARIA_128_GCM_SHA256 0xC06E /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_PSK_WITH_ARIA_256_GCM_SHA384 0xC06F /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_PSK_WITH_ARIA_128_CBC_SHA256 0xC070 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_PSK_WITH_ARIA_256_CBC_SHA384 0xC071 /**< TLS 1.2 */ - -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_128_CBC_SHA256 0xC072 -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_256_CBC_SHA384 0xC073 -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_128_CBC_SHA256 0xC074 -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_256_CBC_SHA384 0xC075 -#define MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_128_CBC_SHA256 0xC076 -#define MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_256_CBC_SHA384 0xC077 -#define MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_128_CBC_SHA256 0xC078 -#define MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_256_CBC_SHA384 0xC079 - -#define MBEDTLS_TLS_RSA_WITH_CAMELLIA_128_GCM_SHA256 0xC07A /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_WITH_CAMELLIA_256_GCM_SHA384 0xC07B /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_128_GCM_SHA256 0xC07C /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_CAMELLIA_256_GCM_SHA384 0xC07D /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_128_GCM_SHA256 0xC086 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_CAMELLIA_256_GCM_SHA384 0xC087 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_128_GCM_SHA256 0xC088 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_ECDSA_WITH_CAMELLIA_256_GCM_SHA384 0xC089 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_128_GCM_SHA256 0xC08A /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_CAMELLIA_256_GCM_SHA384 0xC08B /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_128_GCM_SHA256 0xC08C /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDH_RSA_WITH_CAMELLIA_256_GCM_SHA384 0xC08D /**< TLS 1.2 */ - -#define MBEDTLS_TLS_PSK_WITH_CAMELLIA_128_GCM_SHA256 0xC08E /**< TLS 1.2 */ -#define MBEDTLS_TLS_PSK_WITH_CAMELLIA_256_GCM_SHA384 0xC08F /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_128_GCM_SHA256 0xC090 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_256_GCM_SHA384 0xC091 /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_128_GCM_SHA256 0xC092 /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_256_GCM_SHA384 0xC093 /**< TLS 1.2 */ - -#define MBEDTLS_TLS_PSK_WITH_CAMELLIA_128_CBC_SHA256 0xC094 -#define MBEDTLS_TLS_PSK_WITH_CAMELLIA_256_CBC_SHA384 0xC095 -#define MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_128_CBC_SHA256 0xC096 -#define MBEDTLS_TLS_DHE_PSK_WITH_CAMELLIA_256_CBC_SHA384 0xC097 -#define MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_128_CBC_SHA256 0xC098 -#define MBEDTLS_TLS_RSA_PSK_WITH_CAMELLIA_256_CBC_SHA384 0xC099 -#define MBEDTLS_TLS_ECDHE_PSK_WITH_CAMELLIA_128_CBC_SHA256 0xC09A -#define MBEDTLS_TLS_ECDHE_PSK_WITH_CAMELLIA_256_CBC_SHA384 0xC09B - -#define MBEDTLS_TLS_RSA_WITH_AES_128_CCM 0xC09C /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_WITH_AES_256_CCM 0xC09D /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CCM 0xC09E /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CCM 0xC09F /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_WITH_AES_128_CCM_8 0xC0A0 /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_WITH_AES_256_CCM_8 0xC0A1 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_AES_128_CCM_8 0xC0A2 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_AES_256_CCM_8 0xC0A3 /**< TLS 1.2 */ -#define MBEDTLS_TLS_PSK_WITH_AES_128_CCM 0xC0A4 /**< TLS 1.2 */ -#define MBEDTLS_TLS_PSK_WITH_AES_256_CCM 0xC0A5 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CCM 0xC0A6 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CCM 0xC0A7 /**< TLS 1.2 */ -#define MBEDTLS_TLS_PSK_WITH_AES_128_CCM_8 0xC0A8 /**< TLS 1.2 */ -#define MBEDTLS_TLS_PSK_WITH_AES_256_CCM_8 0xC0A9 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_AES_128_CCM_8 0xC0AA /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_AES_256_CCM_8 0xC0AB /**< TLS 1.2 */ -/* The last two are named with PSK_DHE in the RFC, which looks like a typo */ - -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CCM 0xC0AC /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CCM 0xC0AD /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_128_CCM_8 0xC0AE /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_AES_256_CCM_8 0xC0AF /**< TLS 1.2 */ - -#define MBEDTLS_TLS_ECJPAKE_WITH_AES_128_CCM_8 0xC0FF /**< experimental */ - -/* RFC 7905 */ -#define MBEDTLS_TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256 0xCCA8 /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256 0xCCA9 /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_RSA_WITH_CHACHA20_POLY1305_SHA256 0xCCAA /**< TLS 1.2 */ -#define MBEDTLS_TLS_PSK_WITH_CHACHA20_POLY1305_SHA256 0xCCAB /**< TLS 1.2 */ -#define MBEDTLS_TLS_ECDHE_PSK_WITH_CHACHA20_POLY1305_SHA256 0xCCAC /**< TLS 1.2 */ -#define MBEDTLS_TLS_DHE_PSK_WITH_CHACHA20_POLY1305_SHA256 0xCCAD /**< TLS 1.2 */ -#define MBEDTLS_TLS_RSA_PSK_WITH_CHACHA20_POLY1305_SHA256 0xCCAE /**< TLS 1.2 */ - -/* Reminder: update mbedtls_ssl_premaster_secret when adding a new key exchange. - * Reminder: update MBEDTLS_KEY_EXCHANGE__xxx below - */ -typedef enum { - MBEDTLS_KEY_EXCHANGE_NONE = 0, - MBEDTLS_KEY_EXCHANGE_RSA, - MBEDTLS_KEY_EXCHANGE_DHE_RSA, - MBEDTLS_KEY_EXCHANGE_ECDHE_RSA, - MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA, - MBEDTLS_KEY_EXCHANGE_PSK, - MBEDTLS_KEY_EXCHANGE_DHE_PSK, - MBEDTLS_KEY_EXCHANGE_RSA_PSK, - MBEDTLS_KEY_EXCHANGE_ECDHE_PSK, - MBEDTLS_KEY_EXCHANGE_ECDH_RSA, - MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA, - MBEDTLS_KEY_EXCHANGE_ECJPAKE, -} mbedtls_key_exchange_type_t; - -/* Key exchanges using a certificate */ -#if defined(MBEDTLS_KEY_EXCHANGE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED) -#define MBEDTLS_KEY_EXCHANGE_WITH_CERT_ENABLED -#endif - -/* Key exchanges allowing client certificate requests */ -#if defined(MBEDTLS_KEY_EXCHANGE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED) -#define MBEDTLS_KEY_EXCHANGE_CERT_REQ_ALLOWED_ENABLED -#endif - -/* Key exchanges involving server signature in ServerKeyExchange */ -#if defined(MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED) -#define MBEDTLS_KEY_EXCHANGE_WITH_SERVER_SIGNATURE_ENABLED -#endif - -/* Key exchanges using ECDH */ -#if defined(MBEDTLS_KEY_EXCHANGE_ECDH_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA_ENABLED) -#define MBEDTLS_KEY_EXCHANGE_SOME_ECDH_ENABLED -#endif - -/* Key exchanges that don't involve ephemeral keys */ -#if defined(MBEDTLS_KEY_EXCHANGE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_SOME_ECDH_ENABLED) -#define MBEDTLS_KEY_EXCHANGE_SOME_NON_PFS_ENABLED -#endif - -/* Key exchanges that involve ephemeral keys */ -#if defined(MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_DHE_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECJPAKE_ENABLED) -#define MBEDTLS_KEY_EXCHANGE_SOME_PFS_ENABLED -#endif - -/* Key exchanges using a PSK */ -#if defined(MBEDTLS_KEY_EXCHANGE_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_RSA_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_DHE_PSK_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED) -#define MBEDTLS_KEY_EXCHANGE_SOME_PSK_ENABLED -#endif - -/* Key exchanges using DHE */ -#if defined(MBEDTLS_KEY_EXCHANGE_DHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_DHE_PSK_ENABLED) -#define MBEDTLS_KEY_EXCHANGE_SOME_DHE_ENABLED -#endif - -/* Key exchanges using ECDHE */ -#if defined(MBEDTLS_KEY_EXCHANGE_ECDHE_RSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA_ENABLED) || \ - defined(MBEDTLS_KEY_EXCHANGE_ECDHE_PSK_ENABLED) -#define MBEDTLS_KEY_EXCHANGE_SOME_ECDHE_ENABLED -#endif - -typedef struct mbedtls_ssl_ciphersuite_t mbedtls_ssl_ciphersuite_t; - -#define MBEDTLS_CIPHERSUITE_WEAK 0x01 /**< Weak ciphersuite flag */ -#define MBEDTLS_CIPHERSUITE_SHORT_TAG 0x02 /**< Short authentication tag, - eg for CCM_8 */ -#define MBEDTLS_CIPHERSUITE_NODTLS 0x04 /**< Can't be used with DTLS */ - -/** - * \brief This structure is used for storing ciphersuite information - */ -struct mbedtls_ssl_ciphersuite_t -{ - int MBEDTLS_PRIVATE(id); - const char * MBEDTLS_PRIVATE(name); - - mbedtls_cipher_type_t MBEDTLS_PRIVATE(cipher); - mbedtls_md_type_t MBEDTLS_PRIVATE(mac); - mbedtls_key_exchange_type_t MBEDTLS_PRIVATE(key_exchange); - - int MBEDTLS_PRIVATE(min_major_ver); - int MBEDTLS_PRIVATE(min_minor_ver); - int MBEDTLS_PRIVATE(max_major_ver); - int MBEDTLS_PRIVATE(max_minor_ver); - - unsigned char MBEDTLS_PRIVATE(flags); -}; - -const int *mbedtls_ssl_list_ciphersuites( void ); - -const mbedtls_ssl_ciphersuite_t *mbedtls_ssl_ciphersuite_from_string( const char *ciphersuite_name ); -const mbedtls_ssl_ciphersuite_t *mbedtls_ssl_ciphersuite_from_id( int ciphersuite_id ); - -#if defined(MBEDTLS_PK_C) -mbedtls_pk_type_t mbedtls_ssl_get_ciphersuite_sig_pk_alg( const mbedtls_ssl_ciphersuite_t *info ); -mbedtls_pk_type_t mbedtls_ssl_get_ciphersuite_sig_alg( const mbedtls_ssl_ciphersuite_t *info ); -#endif - -int mbedtls_ssl_ciphersuite_uses_ec( const mbedtls_ssl_ciphersuite_t *info ); -int mbedtls_ssl_ciphersuite_uses_psk( const mbedtls_ssl_ciphersuite_t *info ); - -#if defined(MBEDTLS_KEY_EXCHANGE_SOME_PFS_ENABLED) -static inline int mbedtls_ssl_ciphersuite_has_pfs( const mbedtls_ssl_ciphersuite_t *info ) -{ - switch( info->MBEDTLS_PRIVATE(key_exchange) ) - { - case MBEDTLS_KEY_EXCHANGE_DHE_RSA: - case MBEDTLS_KEY_EXCHANGE_DHE_PSK: - case MBEDTLS_KEY_EXCHANGE_ECDHE_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDHE_PSK: - case MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA: - case MBEDTLS_KEY_EXCHANGE_ECJPAKE: - return( 1 ); - - default: - return( 0 ); - } -} -#endif /* MBEDTLS_KEY_EXCHANGE_SOME_PFS_ENABLED */ - -#if defined(MBEDTLS_KEY_EXCHANGE_SOME_NON_PFS_ENABLED) -static inline int mbedtls_ssl_ciphersuite_no_pfs( const mbedtls_ssl_ciphersuite_t *info ) -{ - switch( info->MBEDTLS_PRIVATE(key_exchange) ) - { - case MBEDTLS_KEY_EXCHANGE_ECDH_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA: - case MBEDTLS_KEY_EXCHANGE_RSA: - case MBEDTLS_KEY_EXCHANGE_PSK: - case MBEDTLS_KEY_EXCHANGE_RSA_PSK: - return( 1 ); - - default: - return( 0 ); - } -} -#endif /* MBEDTLS_KEY_EXCHANGE_SOME_NON_PFS_ENABLED */ - -#if defined(MBEDTLS_KEY_EXCHANGE_SOME_ECDH_ENABLED) -static inline int mbedtls_ssl_ciphersuite_uses_ecdh( const mbedtls_ssl_ciphersuite_t *info ) -{ - switch( info->MBEDTLS_PRIVATE(key_exchange) ) - { - case MBEDTLS_KEY_EXCHANGE_ECDH_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA: - return( 1 ); - - default: - return( 0 ); - } -} -#endif /* MBEDTLS_KEY_EXCHANGE_SOME_ECDH_ENABLED */ - -static inline int mbedtls_ssl_ciphersuite_cert_req_allowed( const mbedtls_ssl_ciphersuite_t *info ) -{ - switch( info->MBEDTLS_PRIVATE(key_exchange) ) - { - case MBEDTLS_KEY_EXCHANGE_RSA: - case MBEDTLS_KEY_EXCHANGE_DHE_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDH_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDHE_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA: - case MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA: - return( 1 ); - - default: - return( 0 ); - } -} - -static inline int mbedtls_ssl_ciphersuite_uses_srv_cert( const mbedtls_ssl_ciphersuite_t *info ) -{ - switch( info->MBEDTLS_PRIVATE(key_exchange) ) - { - case MBEDTLS_KEY_EXCHANGE_RSA: - case MBEDTLS_KEY_EXCHANGE_RSA_PSK: - case MBEDTLS_KEY_EXCHANGE_DHE_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDH_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDHE_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDH_ECDSA: - case MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA: - return( 1 ); - - default: - return( 0 ); - } -} - -#if defined(MBEDTLS_KEY_EXCHANGE_SOME_DHE_ENABLED) -static inline int mbedtls_ssl_ciphersuite_uses_dhe( const mbedtls_ssl_ciphersuite_t *info ) -{ - switch( info->MBEDTLS_PRIVATE(key_exchange) ) - { - case MBEDTLS_KEY_EXCHANGE_DHE_RSA: - case MBEDTLS_KEY_EXCHANGE_DHE_PSK: - return( 1 ); - - default: - return( 0 ); - } -} -#endif /* MBEDTLS_KEY_EXCHANGE_SOME_DHE_ENABLED) */ - -#if defined(MBEDTLS_KEY_EXCHANGE_SOME_ECDHE_ENABLED) -static inline int mbedtls_ssl_ciphersuite_uses_ecdhe( const mbedtls_ssl_ciphersuite_t *info ) -{ - switch( info->MBEDTLS_PRIVATE(key_exchange) ) - { - case MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA: - case MBEDTLS_KEY_EXCHANGE_ECDHE_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDHE_PSK: - return( 1 ); - - default: - return( 0 ); - } -} -#endif /* MBEDTLS_KEY_EXCHANGE_SOME_ECDHE_ENABLED) */ - -#if defined(MBEDTLS_KEY_EXCHANGE_WITH_SERVER_SIGNATURE_ENABLED) -static inline int mbedtls_ssl_ciphersuite_uses_server_signature( const mbedtls_ssl_ciphersuite_t *info ) -{ - switch( info->MBEDTLS_PRIVATE(key_exchange) ) - { - case MBEDTLS_KEY_EXCHANGE_DHE_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDHE_RSA: - case MBEDTLS_KEY_EXCHANGE_ECDHE_ECDSA: - return( 1 ); - - default: - return( 0 ); - } -} -#endif /* MBEDTLS_KEY_EXCHANGE_WITH_SERVER_SIGNATURE_ENABLED */ - -#ifdef __cplusplus -} -#endif - -#endif /* ssl_ciphersuites.h */ diff --git a/windows/mbedtls/ssl_cookie.h b/windows/mbedtls/ssl_cookie.h deleted file mode 100644 index 86698b05..00000000 --- a/windows/mbedtls/ssl_cookie.h +++ /dev/null @@ -1,110 +0,0 @@ -/** - * \file ssl_cookie.h - * - * \brief DTLS cookie callbacks implementation - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_SSL_COOKIE_H -#define MBEDTLS_SSL_COOKIE_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/ssl.h" - -#if defined(MBEDTLS_THREADING_C) -#include "mbedtls/threading.h" -#endif - -/** - * \name SECTION: Module settings - * - * The configuration options you can set for this module are in this section. - * Either change them in mbedtls_config.h or define them on the compiler command line. - * \{ - */ -#ifndef MBEDTLS_SSL_COOKIE_TIMEOUT -#define MBEDTLS_SSL_COOKIE_TIMEOUT 60 /**< Default expiration delay of DTLS cookies, in seconds if HAVE_TIME, or in number of cookies issued */ -#endif - -/* \} name SECTION: Module settings */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief Context for the default cookie functions. - */ -typedef struct mbedtls_ssl_cookie_ctx -{ - mbedtls_md_context_t MBEDTLS_PRIVATE(hmac_ctx); /*!< context for the HMAC portion */ -#if !defined(MBEDTLS_HAVE_TIME) - unsigned long MBEDTLS_PRIVATE(serial); /*!< serial number for expiration */ -#endif - unsigned long MBEDTLS_PRIVATE(timeout); /*!< timeout delay, in seconds if HAVE_TIME, - or in number of tickets issued */ - -#if defined(MBEDTLS_THREADING_C) - mbedtls_threading_mutex_t MBEDTLS_PRIVATE(mutex); -#endif -} mbedtls_ssl_cookie_ctx; - -/** - * \brief Initialize cookie context - */ -void mbedtls_ssl_cookie_init( mbedtls_ssl_cookie_ctx *ctx ); - -/** - * \brief Setup cookie context (generate keys) - */ -int mbedtls_ssl_cookie_setup( mbedtls_ssl_cookie_ctx *ctx, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -/** - * \brief Set expiration delay for cookies - * (Default MBEDTLS_SSL_COOKIE_TIMEOUT) - * - * \param ctx Cookie contex - * \param delay Delay, in seconds if HAVE_TIME, or in number of cookies - * issued in the meantime. - * 0 to disable expiration (NOT recommended) - */ -void mbedtls_ssl_cookie_set_timeout( mbedtls_ssl_cookie_ctx *ctx, unsigned long delay ); - -/** - * \brief Free cookie context - */ -void mbedtls_ssl_cookie_free( mbedtls_ssl_cookie_ctx *ctx ); - -/** - * \brief Generate cookie, see \c mbedtls_ssl_cookie_write_t - */ -mbedtls_ssl_cookie_write_t mbedtls_ssl_cookie_write; - -/** - * \brief Verify cookie, see \c mbedtls_ssl_cookie_write_t - */ -mbedtls_ssl_cookie_check_t mbedtls_ssl_cookie_check; - -#ifdef __cplusplus -} -#endif - -#endif /* ssl_cookie.h */ diff --git a/windows/mbedtls/ssl_ticket.h b/windows/mbedtls/ssl_ticket.h deleted file mode 100644 index 0f4117d3..00000000 --- a/windows/mbedtls/ssl_ticket.h +++ /dev/null @@ -1,137 +0,0 @@ -/** - * \file ssl_ticket.h - * - * \brief TLS server ticket callbacks implementation - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_SSL_TICKET_H -#define MBEDTLS_SSL_TICKET_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -/* - * This implementation of the session ticket callbacks includes key - * management, rotating the keys periodically in order to preserve forward - * secrecy, when MBEDTLS_HAVE_TIME is defined. - */ - -#include "mbedtls/ssl.h" -#include "mbedtls/cipher.h" - -#if defined(MBEDTLS_THREADING_C) -#include "mbedtls/threading.h" -#endif - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \brief Information for session ticket protection - */ -typedef struct mbedtls_ssl_ticket_key -{ - unsigned char MBEDTLS_PRIVATE(name)[4]; /*!< random key identifier */ - uint32_t MBEDTLS_PRIVATE(generation_time); /*!< key generation timestamp (seconds) */ - mbedtls_cipher_context_t MBEDTLS_PRIVATE(ctx); /*!< context for auth enc/decryption */ -} -mbedtls_ssl_ticket_key; - -/** - * \brief Context for session ticket handling functions - */ -typedef struct mbedtls_ssl_ticket_context -{ - mbedtls_ssl_ticket_key MBEDTLS_PRIVATE(keys)[2]; /*!< ticket protection keys */ - unsigned char MBEDTLS_PRIVATE(active); /*!< index of the currently active key */ - - uint32_t MBEDTLS_PRIVATE(ticket_lifetime); /*!< lifetime of tickets in seconds */ - - /** Callback for getting (pseudo-)random numbers */ - int (*MBEDTLS_PRIVATE(f_rng))(void *, unsigned char *, size_t); - void *MBEDTLS_PRIVATE(p_rng); /*!< context for the RNG function */ - -#if defined(MBEDTLS_THREADING_C) - mbedtls_threading_mutex_t MBEDTLS_PRIVATE(mutex); -#endif -} -mbedtls_ssl_ticket_context; - -/** - * \brief Initialize a ticket context. - * (Just make it ready for mbedtls_ssl_ticket_setup() - * or mbedtls_ssl_ticket_free().) - * - * \param ctx Context to be initialized - */ -void mbedtls_ssl_ticket_init( mbedtls_ssl_ticket_context *ctx ); - -/** - * \brief Prepare context to be actually used - * - * \param ctx Context to be set up - * \param f_rng RNG callback function (mandatory) - * \param p_rng RNG callback context - * \param cipher AEAD cipher to use for ticket protection. - * Recommended value: MBEDTLS_CIPHER_AES_256_GCM. - * \param lifetime Tickets lifetime in seconds - * Recommended value: 86400 (one day). - * - * \note It is highly recommended to select a cipher that is at - * least as strong as the strongest ciphersuite - * supported. Usually that means a 256-bit key. - * - * \note The lifetime of the keys is twice the lifetime of tickets. - * It is recommended to pick a reasonnable lifetime so as not - * to negate the benefits of forward secrecy. - * - * \return 0 if successful, - * or a specific MBEDTLS_ERR_XXX error code - */ -int mbedtls_ssl_ticket_setup( mbedtls_ssl_ticket_context *ctx, - int (*f_rng)(void *, unsigned char *, size_t), void *p_rng, - mbedtls_cipher_type_t cipher, - uint32_t lifetime ); - -/** - * \brief Implementation of the ticket write callback - * - * \note See \c mbedtls_ssl_ticket_write_t for description - */ -mbedtls_ssl_ticket_write_t mbedtls_ssl_ticket_write; - -/** - * \brief Implementation of the ticket parse callback - * - * \note See \c mbedtls_ssl_ticket_parse_t for description - */ -mbedtls_ssl_ticket_parse_t mbedtls_ssl_ticket_parse; - -/** - * \brief Free a context's content and zeroize it. - * - * \param ctx Context to be cleaned up - */ -void mbedtls_ssl_ticket_free( mbedtls_ssl_ticket_context *ctx ); - -#ifdef __cplusplus -} -#endif - -#endif /* ssl_ticket.h */ diff --git a/windows/mbedtls/threading.h b/windows/mbedtls/threading.h deleted file mode 100644 index 96cadc4d..00000000 --- a/windows/mbedtls/threading.h +++ /dev/null @@ -1,116 +0,0 @@ -/** - * \file threading.h - * - * \brief Threading abstraction layer - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_THREADING_H -#define MBEDTLS_THREADING_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include - -#ifdef __cplusplus -extern "C" { -#endif - -#define MBEDTLS_ERR_THREADING_BAD_INPUT_DATA -0x001C /**< Bad input parameters to function. */ -#define MBEDTLS_ERR_THREADING_MUTEX_ERROR -0x001E /**< Locking / unlocking / free failed with error code. */ - -#if defined(MBEDTLS_THREADING_PTHREAD) -#include -typedef struct mbedtls_threading_mutex_t -{ - pthread_mutex_t MBEDTLS_PRIVATE(mutex); - /* is_valid is 0 after a failed init or a free, and nonzero after a - * successful init. This field is not considered part of the public - * API of Mbed TLS and may change without notice. */ - char MBEDTLS_PRIVATE(is_valid); -} mbedtls_threading_mutex_t; -#endif - -#if defined(MBEDTLS_THREADING_ALT) -/* You should define the mbedtls_threading_mutex_t type in your header */ -#include "threading_alt.h" - -/** - * \brief Set your alternate threading implementation function - * pointers and initialize global mutexes. If used, this - * function must be called once in the main thread before any - * other mbed TLS function is called, and - * mbedtls_threading_free_alt() must be called once in the main - * thread after all other mbed TLS functions. - * - * \note mutex_init() and mutex_free() don't return a status code. - * If mutex_init() fails, it should leave its argument (the - * mutex) in a state such that mutex_lock() will fail when - * called with this argument. - * - * \param mutex_init the init function implementation - * \param mutex_free the free function implementation - * \param mutex_lock the lock function implementation - * \param mutex_unlock the unlock function implementation - */ -void mbedtls_threading_set_alt( void (*mutex_init)( mbedtls_threading_mutex_t * ), - void (*mutex_free)( mbedtls_threading_mutex_t * ), - int (*mutex_lock)( mbedtls_threading_mutex_t * ), - int (*mutex_unlock)( mbedtls_threading_mutex_t * ) ); - -/** - * \brief Free global mutexes. - */ -void mbedtls_threading_free_alt( void ); -#endif /* MBEDTLS_THREADING_ALT */ - -#if defined(MBEDTLS_THREADING_C) -/* - * The function pointers for mutex_init, mutex_free, mutex_ and mutex_unlock - * - * All these functions are expected to work or the result will be undefined. - */ -extern void (*mbedtls_mutex_init)( mbedtls_threading_mutex_t *mutex ); -extern void (*mbedtls_mutex_free)( mbedtls_threading_mutex_t *mutex ); -extern int (*mbedtls_mutex_lock)( mbedtls_threading_mutex_t *mutex ); -extern int (*mbedtls_mutex_unlock)( mbedtls_threading_mutex_t *mutex ); - -/* - * Global mutexes - */ -#if defined(MBEDTLS_FS_IO) -extern mbedtls_threading_mutex_t mbedtls_threading_readdir_mutex; -#endif - -#if defined(MBEDTLS_HAVE_TIME_DATE) && !defined(MBEDTLS_PLATFORM_GMTIME_R_ALT) -/* This mutex may or may not be used in the default definition of - * mbedtls_platform_gmtime_r(), but in order to determine that, - * we need to check POSIX features, hence modify _POSIX_C_SOURCE. - * With the current approach, this declaration is orphaned, lacking - * an accompanying definition, in case mbedtls_platform_gmtime_r() - * doesn't need it, but that's not a problem. */ -extern mbedtls_threading_mutex_t mbedtls_threading_gmtime_mutex; -#endif /* MBEDTLS_HAVE_TIME_DATE && !MBEDTLS_PLATFORM_GMTIME_R_ALT */ - -#endif /* MBEDTLS_THREADING_C */ - -#ifdef __cplusplus -} -#endif - -#endif /* threading.h */ diff --git a/windows/mbedtls/timing.h b/windows/mbedtls/timing.h deleted file mode 100644 index 25db1c6a..00000000 --- a/windows/mbedtls/timing.h +++ /dev/null @@ -1,97 +0,0 @@ -/** - * \file timing.h - * - * \brief Portable interface to timeouts and to the CPU cycle counter - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_TIMING_H -#define MBEDTLS_TIMING_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include - -#ifdef __cplusplus -extern "C" { -#endif - -#if !defined(MBEDTLS_TIMING_ALT) -// Regular implementation -// - -/** - * \brief timer structure - */ -struct mbedtls_timing_hr_time -{ - unsigned char MBEDTLS_PRIVATE(opaque)[32]; -}; - -/** - * \brief Context for mbedtls_timing_set/get_delay() - */ -typedef struct mbedtls_timing_delay_context -{ - struct mbedtls_timing_hr_time MBEDTLS_PRIVATE(timer); - uint32_t MBEDTLS_PRIVATE(int_ms); - uint32_t MBEDTLS_PRIVATE(fin_ms); -} mbedtls_timing_delay_context; - -#else /* MBEDTLS_TIMING_ALT */ -#include "timing_alt.h" -#endif /* MBEDTLS_TIMING_ALT */ - -/* Internal use */ -unsigned long mbedtls_timing_get_timer( struct mbedtls_timing_hr_time *val, int reset ); - -/** - * \brief Set a pair of delays to watch - * (See \c mbedtls_timing_get_delay().) - * - * \param data Pointer to timing data. - * Must point to a valid \c mbedtls_timing_delay_context struct. - * \param int_ms First (intermediate) delay in milliseconds. - * The effect if int_ms > fin_ms is unspecified. - * \param fin_ms Second (final) delay in milliseconds. - * Pass 0 to cancel the current delay. - * - * \note To set a single delay, either use \c mbedtls_timing_set_timer - * directly or use this function with int_ms == fin_ms. - */ -void mbedtls_timing_set_delay( void *data, uint32_t int_ms, uint32_t fin_ms ); - -/** - * \brief Get the status of delays - * (Memory helper: number of delays passed.) - * - * \param data Pointer to timing data - * Must point to a valid \c mbedtls_timing_delay_context struct. - * - * \return -1 if cancelled (fin_ms = 0), - * 0 if none of the delays are passed, - * 1 if only the intermediate delay is passed, - * 2 if the final delay is passed. - */ -int mbedtls_timing_get_delay( void *data ); - -#ifdef __cplusplus -} -#endif - -#endif /* timing.h */ diff --git a/windows/mbedtls/version.h b/windows/mbedtls/version.h deleted file mode 100644 index 773da4a6..00000000 --- a/windows/mbedtls/version.h +++ /dev/null @@ -1,90 +0,0 @@ -/** - * \file version.h - * - * \brief Run-time version information - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -/* - * This set of run-time variables can be used to determine the version number of - * the Mbed TLS library used. Compile-time version defines for the same can be - * found in build_info.h - */ -#ifndef MBEDTLS_VERSION_H -#define MBEDTLS_VERSION_H - -#include "mbedtls/build_info.h" - -#if defined(MBEDTLS_VERSION_C) - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * Get the version number. - * - * \return The constructed version number in the format - * MMNNPP00 (Major, Minor, Patch). - */ -unsigned int mbedtls_version_get_number( void ); - -/** - * Get the version string ("x.y.z"). - * - * \param string The string that will receive the value. - * (Should be at least 9 bytes in size) - */ -void mbedtls_version_get_string( char *string ); - -/** - * Get the full version string ("mbed TLS x.y.z"). - * - * \param string The string that will receive the value. The mbed TLS version - * string will use 18 bytes AT MOST including a terminating - * null byte. - * (So the buffer should be at least 18 bytes to receive this - * version string). - */ -void mbedtls_version_get_string_full( char *string ); - -/** - * \brief Check if support for a feature was compiled into this - * mbed TLS binary. This allows you to see at runtime if the - * library was for instance compiled with or without - * Multi-threading support. - * - * \note only checks against defines in the sections "System - * support", "mbed TLS modules" and "mbed TLS feature - * support" in mbedtls_config.h - * - * \param feature The string for the define to check (e.g. "MBEDTLS_AES_C") - * - * \return 0 if the feature is present, - * -1 if the feature is not present and - * -2 if support for feature checking as a whole was not - * compiled in. - */ -int mbedtls_version_check_feature( const char *feature ); - -#ifdef __cplusplus -} -#endif - -#endif /* MBEDTLS_VERSION_C */ - -#endif /* version.h */ diff --git a/windows/mbedtls/x509.h b/windows/mbedtls/x509.h deleted file mode 100644 index 5be4286e..00000000 --- a/windows/mbedtls/x509.h +++ /dev/null @@ -1,347 +0,0 @@ -/** - * \file x509.h - * - * \brief X.509 generic defines and structures - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_X509_H -#define MBEDTLS_X509_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/asn1.h" -#include "mbedtls/pk.h" - -#if defined(MBEDTLS_RSA_C) -#include "mbedtls/rsa.h" -#endif - -/** - * \addtogroup x509_module - * \{ - */ - -#if !defined(MBEDTLS_X509_MAX_INTERMEDIATE_CA) -/** - * Maximum number of intermediate CAs in a verification chain. - * That is, maximum length of the chain, excluding the end-entity certificate - * and the trusted root certificate. - * - * Set this to a low value to prevent an adversary from making you waste - * resources verifying an overlong certificate chain. - */ -#define MBEDTLS_X509_MAX_INTERMEDIATE_CA 8 -#endif - -/** - * \name X509 Error codes - * \{ - */ -#define MBEDTLS_ERR_X509_FEATURE_UNAVAILABLE -0x2080 /**< Unavailable feature, e.g. RSA hashing/encryption combination. */ -#define MBEDTLS_ERR_X509_UNKNOWN_OID -0x2100 /**< Requested OID is unknown. */ -#define MBEDTLS_ERR_X509_INVALID_FORMAT -0x2180 /**< The CRT/CRL/CSR format is invalid, e.g. different type expected. */ -#define MBEDTLS_ERR_X509_INVALID_VERSION -0x2200 /**< The CRT/CRL/CSR version element is invalid. */ -#define MBEDTLS_ERR_X509_INVALID_SERIAL -0x2280 /**< The serial tag or value is invalid. */ -#define MBEDTLS_ERR_X509_INVALID_ALG -0x2300 /**< The algorithm tag or value is invalid. */ -#define MBEDTLS_ERR_X509_INVALID_NAME -0x2380 /**< The name tag or value is invalid. */ -#define MBEDTLS_ERR_X509_INVALID_DATE -0x2400 /**< The date tag or value is invalid. */ -#define MBEDTLS_ERR_X509_INVALID_SIGNATURE -0x2480 /**< The signature tag or value invalid. */ -#define MBEDTLS_ERR_X509_INVALID_EXTENSIONS -0x2500 /**< The extension tag or value is invalid. */ -#define MBEDTLS_ERR_X509_UNKNOWN_VERSION -0x2580 /**< CRT/CRL/CSR has an unsupported version number. */ -#define MBEDTLS_ERR_X509_UNKNOWN_SIG_ALG -0x2600 /**< Signature algorithm (oid) is unsupported. */ -#define MBEDTLS_ERR_X509_SIG_MISMATCH -0x2680 /**< Signature algorithms do not match. (see \c ::mbedtls_x509_crt sig_oid) */ -#define MBEDTLS_ERR_X509_CERT_VERIFY_FAILED -0x2700 /**< Certificate verification failed, e.g. CRL, CA or signature check failed. */ -#define MBEDTLS_ERR_X509_CERT_UNKNOWN_FORMAT -0x2780 /**< Format not recognized as DER or PEM. */ -#define MBEDTLS_ERR_X509_BAD_INPUT_DATA -0x2800 /**< Input invalid. */ -#define MBEDTLS_ERR_X509_ALLOC_FAILED -0x2880 /**< Allocation of memory failed. */ -#define MBEDTLS_ERR_X509_FILE_IO_ERROR -0x2900 /**< Read/write of file failed. */ -#define MBEDTLS_ERR_X509_BUFFER_TOO_SMALL -0x2980 /**< Destination buffer is too small. */ -#define MBEDTLS_ERR_X509_FATAL_ERROR -0x3000 /**< A fatal error occurred, eg the chain is too long or the vrfy callback failed. */ -/* \} name */ - -/** - * \name X509 Verify codes - * \{ - */ -/* Reminder: update x509_crt_verify_strings[] in library/x509_crt.c */ -#define MBEDTLS_X509_BADCERT_EXPIRED 0x01 /**< The certificate validity has expired. */ -#define MBEDTLS_X509_BADCERT_REVOKED 0x02 /**< The certificate has been revoked (is on a CRL). */ -#define MBEDTLS_X509_BADCERT_CN_MISMATCH 0x04 /**< The certificate Common Name (CN) does not match with the expected CN. */ -#define MBEDTLS_X509_BADCERT_NOT_TRUSTED 0x08 /**< The certificate is not correctly signed by the trusted CA. */ -#define MBEDTLS_X509_BADCRL_NOT_TRUSTED 0x10 /**< The CRL is not correctly signed by the trusted CA. */ -#define MBEDTLS_X509_BADCRL_EXPIRED 0x20 /**< The CRL is expired. */ -#define MBEDTLS_X509_BADCERT_MISSING 0x40 /**< Certificate was missing. */ -#define MBEDTLS_X509_BADCERT_SKIP_VERIFY 0x80 /**< Certificate verification was skipped. */ -#define MBEDTLS_X509_BADCERT_OTHER 0x0100 /**< Other reason (can be used by verify callback) */ -#define MBEDTLS_X509_BADCERT_FUTURE 0x0200 /**< The certificate validity starts in the future. */ -#define MBEDTLS_X509_BADCRL_FUTURE 0x0400 /**< The CRL is from the future */ -#define MBEDTLS_X509_BADCERT_KEY_USAGE 0x0800 /**< Usage does not match the keyUsage extension. */ -#define MBEDTLS_X509_BADCERT_EXT_KEY_USAGE 0x1000 /**< Usage does not match the extendedKeyUsage extension. */ -#define MBEDTLS_X509_BADCERT_NS_CERT_TYPE 0x2000 /**< Usage does not match the nsCertType extension. */ -#define MBEDTLS_X509_BADCERT_BAD_MD 0x4000 /**< The certificate is signed with an unacceptable hash. */ -#define MBEDTLS_X509_BADCERT_BAD_PK 0x8000 /**< The certificate is signed with an unacceptable PK alg (eg RSA vs ECDSA). */ -#define MBEDTLS_X509_BADCERT_BAD_KEY 0x010000 /**< The certificate is signed with an unacceptable key (eg bad curve, RSA too short). */ -#define MBEDTLS_X509_BADCRL_BAD_MD 0x020000 /**< The CRL is signed with an unacceptable hash. */ -#define MBEDTLS_X509_BADCRL_BAD_PK 0x040000 /**< The CRL is signed with an unacceptable PK alg (eg RSA vs ECDSA). */ -#define MBEDTLS_X509_BADCRL_BAD_KEY 0x080000 /**< The CRL is signed with an unacceptable key (eg bad curve, RSA too short). */ - -/* \} name */ -/* \} addtogroup x509_module */ - -/* - * X.509 v3 Subject Alternative Name types. - * otherName [0] OtherName, - * rfc822Name [1] IA5String, - * dNSName [2] IA5String, - * x400Address [3] ORAddress, - * directoryName [4] Name, - * ediPartyName [5] EDIPartyName, - * uniformResourceIdentifier [6] IA5String, - * iPAddress [7] OCTET STRING, - * registeredID [8] OBJECT IDENTIFIER - */ -#define MBEDTLS_X509_SAN_OTHER_NAME 0 -#define MBEDTLS_X509_SAN_RFC822_NAME 1 -#define MBEDTLS_X509_SAN_DNS_NAME 2 -#define MBEDTLS_X509_SAN_X400_ADDRESS_NAME 3 -#define MBEDTLS_X509_SAN_DIRECTORY_NAME 4 -#define MBEDTLS_X509_SAN_EDI_PARTY_NAME 5 -#define MBEDTLS_X509_SAN_UNIFORM_RESOURCE_IDENTIFIER 6 -#define MBEDTLS_X509_SAN_IP_ADDRESS 7 -#define MBEDTLS_X509_SAN_REGISTERED_ID 8 - -/* - * X.509 v3 Key Usage Extension flags - * Reminder: update x509_info_key_usage() when adding new flags. - */ -#define MBEDTLS_X509_KU_DIGITAL_SIGNATURE (0x80) /* bit 0 */ -#define MBEDTLS_X509_KU_NON_REPUDIATION (0x40) /* bit 1 */ -#define MBEDTLS_X509_KU_KEY_ENCIPHERMENT (0x20) /* bit 2 */ -#define MBEDTLS_X509_KU_DATA_ENCIPHERMENT (0x10) /* bit 3 */ -#define MBEDTLS_X509_KU_KEY_AGREEMENT (0x08) /* bit 4 */ -#define MBEDTLS_X509_KU_KEY_CERT_SIGN (0x04) /* bit 5 */ -#define MBEDTLS_X509_KU_CRL_SIGN (0x02) /* bit 6 */ -#define MBEDTLS_X509_KU_ENCIPHER_ONLY (0x01) /* bit 7 */ -#define MBEDTLS_X509_KU_DECIPHER_ONLY (0x8000) /* bit 8 */ - -/* - * Netscape certificate types - * (http://www.mozilla.org/projects/security/pki/nss/tech-notes/tn3.html) - */ - -#define MBEDTLS_X509_NS_CERT_TYPE_SSL_CLIENT (0x80) /* bit 0 */ -#define MBEDTLS_X509_NS_CERT_TYPE_SSL_SERVER (0x40) /* bit 1 */ -#define MBEDTLS_X509_NS_CERT_TYPE_EMAIL (0x20) /* bit 2 */ -#define MBEDTLS_X509_NS_CERT_TYPE_OBJECT_SIGNING (0x10) /* bit 3 */ -#define MBEDTLS_X509_NS_CERT_TYPE_RESERVED (0x08) /* bit 4 */ -#define MBEDTLS_X509_NS_CERT_TYPE_SSL_CA (0x04) /* bit 5 */ -#define MBEDTLS_X509_NS_CERT_TYPE_EMAIL_CA (0x02) /* bit 6 */ -#define MBEDTLS_X509_NS_CERT_TYPE_OBJECT_SIGNING_CA (0x01) /* bit 7 */ - -/* - * X.509 extension types - * - * Comments refer to the status for using certificates. Status can be - * different for writing certificates or reading CRLs or CSRs. - * - * Those are defined in oid.h as oid.c needs them in a data structure. Since - * these were previously defined here, let's have aliases for compatibility. - */ -#define MBEDTLS_X509_EXT_AUTHORITY_KEY_IDENTIFIER MBEDTLS_OID_X509_EXT_AUTHORITY_KEY_IDENTIFIER -#define MBEDTLS_X509_EXT_SUBJECT_KEY_IDENTIFIER MBEDTLS_OID_X509_EXT_SUBJECT_KEY_IDENTIFIER -#define MBEDTLS_X509_EXT_KEY_USAGE MBEDTLS_OID_X509_EXT_KEY_USAGE -#define MBEDTLS_X509_EXT_CERTIFICATE_POLICIES MBEDTLS_OID_X509_EXT_CERTIFICATE_POLICIES -#define MBEDTLS_X509_EXT_POLICY_MAPPINGS MBEDTLS_OID_X509_EXT_POLICY_MAPPINGS -#define MBEDTLS_X509_EXT_SUBJECT_ALT_NAME MBEDTLS_OID_X509_EXT_SUBJECT_ALT_NAME /* Supported (DNS) */ -#define MBEDTLS_X509_EXT_ISSUER_ALT_NAME MBEDTLS_OID_X509_EXT_ISSUER_ALT_NAME -#define MBEDTLS_X509_EXT_SUBJECT_DIRECTORY_ATTRS MBEDTLS_OID_X509_EXT_SUBJECT_DIRECTORY_ATTRS -#define MBEDTLS_X509_EXT_BASIC_CONSTRAINTS MBEDTLS_OID_X509_EXT_BASIC_CONSTRAINTS /* Supported */ -#define MBEDTLS_X509_EXT_NAME_CONSTRAINTS MBEDTLS_OID_X509_EXT_NAME_CONSTRAINTS -#define MBEDTLS_X509_EXT_POLICY_CONSTRAINTS MBEDTLS_OID_X509_EXT_POLICY_CONSTRAINTS -#define MBEDTLS_X509_EXT_EXTENDED_KEY_USAGE MBEDTLS_OID_X509_EXT_EXTENDED_KEY_USAGE -#define MBEDTLS_X509_EXT_CRL_DISTRIBUTION_POINTS MBEDTLS_OID_X509_EXT_CRL_DISTRIBUTION_POINTS -#define MBEDTLS_X509_EXT_INIHIBIT_ANYPOLICY MBEDTLS_OID_X509_EXT_INIHIBIT_ANYPOLICY -#define MBEDTLS_X509_EXT_FRESHEST_CRL MBEDTLS_OID_X509_EXT_FRESHEST_CRL -#define MBEDTLS_X509_EXT_NS_CERT_TYPE MBEDTLS_OID_X509_EXT_NS_CERT_TYPE - -/* - * Storage format identifiers - * Recognized formats: PEM and DER - */ -#define MBEDTLS_X509_FORMAT_DER 1 -#define MBEDTLS_X509_FORMAT_PEM 2 - -#define MBEDTLS_X509_MAX_DN_NAME_SIZE 256 /**< Maximum value size of a DN entry */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \addtogroup x509_module - * \{ */ - -/** - * \name Structures for parsing X.509 certificates, CRLs and CSRs - * \{ - */ - -/** - * Type-length-value structure that allows for ASN1 using DER. - */ -typedef mbedtls_asn1_buf mbedtls_x509_buf; - -/** - * Container for ASN1 bit strings. - */ -typedef mbedtls_asn1_bitstring mbedtls_x509_bitstring; - -/** - * Container for ASN1 named information objects. - * It allows for Relative Distinguished Names (e.g. cn=localhost,ou=code,etc.). - */ -typedef mbedtls_asn1_named_data mbedtls_x509_name; - -/** - * Container for a sequence of ASN.1 items - */ -typedef mbedtls_asn1_sequence mbedtls_x509_sequence; - -/** Container for date and time (precision in seconds). */ -typedef struct mbedtls_x509_time -{ - int MBEDTLS_PRIVATE(year), MBEDTLS_PRIVATE(mon), MBEDTLS_PRIVATE(day); /**< Date. */ - int MBEDTLS_PRIVATE(hour), MBEDTLS_PRIVATE(min), MBEDTLS_PRIVATE(sec); /**< Time. */ -} -mbedtls_x509_time; - -/** \} name Structures for parsing X.509 certificates, CRLs and CSRs */ -/** \} addtogroup x509_module */ - -/** - * \brief Store the certificate DN in printable form into buf; - * no more than size characters will be written. - * - * \param buf Buffer to write to - * \param size Maximum size of buffer - * \param dn The X509 name to represent - * - * \return The length of the string written (not including the - * terminated nul byte), or a negative error code. - */ -int mbedtls_x509_dn_gets( char *buf, size_t size, const mbedtls_x509_name *dn ); - -/** - * \brief Store the certificate serial in printable form into buf; - * no more than size characters will be written. - * - * \param buf Buffer to write to - * \param size Maximum size of buffer - * \param serial The X509 serial to represent - * - * \return The length of the string written (not including the - * terminated nul byte), or a negative error code. - */ -int mbedtls_x509_serial_gets( char *buf, size_t size, const mbedtls_x509_buf *serial ); - -/** - * \brief Check a given mbedtls_x509_time against the system time - * and tell if it's in the past. - * - * \note Intended usage is "if( is_past( valid_to ) ) ERROR". - * Hence the return value of 1 if on internal errors. - * - * \param to mbedtls_x509_time to check - * - * \return 1 if the given time is in the past or an error occurred, - * 0 otherwise. - */ -int mbedtls_x509_time_is_past( const mbedtls_x509_time *to ); - -/** - * \brief Check a given mbedtls_x509_time against the system time - * and tell if it's in the future. - * - * \note Intended usage is "if( is_future( valid_from ) ) ERROR". - * Hence the return value of 1 if on internal errors. - * - * \param from mbedtls_x509_time to check - * - * \return 1 if the given time is in the future or an error occurred, - * 0 otherwise. - */ -int mbedtls_x509_time_is_future( const mbedtls_x509_time *from ); - -/* - * Internal module functions. You probably do not want to use these unless you - * know you do. - */ -int mbedtls_x509_get_name( unsigned char **p, const unsigned char *end, - mbedtls_x509_name *cur ); -int mbedtls_x509_get_alg_null( unsigned char **p, const unsigned char *end, - mbedtls_x509_buf *alg ); -int mbedtls_x509_get_alg( unsigned char **p, const unsigned char *end, - mbedtls_x509_buf *alg, mbedtls_x509_buf *params ); -#if defined(MBEDTLS_X509_RSASSA_PSS_SUPPORT) -int mbedtls_x509_get_rsassa_pss_params( const mbedtls_x509_buf *params, - mbedtls_md_type_t *md_alg, mbedtls_md_type_t *mgf_md, - int *salt_len ); -#endif -int mbedtls_x509_get_sig( unsigned char **p, const unsigned char *end, mbedtls_x509_buf *sig ); -int mbedtls_x509_get_sig_alg( const mbedtls_x509_buf *sig_oid, const mbedtls_x509_buf *sig_params, - mbedtls_md_type_t *md_alg, mbedtls_pk_type_t *pk_alg, - void **sig_opts ); -int mbedtls_x509_get_time( unsigned char **p, const unsigned char *end, - mbedtls_x509_time *t ); -int mbedtls_x509_get_serial( unsigned char **p, const unsigned char *end, - mbedtls_x509_buf *serial ); -int mbedtls_x509_get_ext( unsigned char **p, const unsigned char *end, - mbedtls_x509_buf *ext, int tag ); -#if !defined(MBEDTLS_X509_REMOVE_INFO) -int mbedtls_x509_sig_alg_gets( char *buf, size_t size, const mbedtls_x509_buf *sig_oid, - mbedtls_pk_type_t pk_alg, mbedtls_md_type_t md_alg, - const void *sig_opts ); -#endif -int mbedtls_x509_key_size_helper( char *buf, size_t buf_size, const char *name ); -int mbedtls_x509_string_to_names( mbedtls_asn1_named_data **head, const char *name ); -int mbedtls_x509_set_extension( mbedtls_asn1_named_data **head, const char *oid, size_t oid_len, - int critical, const unsigned char *val, - size_t val_len ); -int mbedtls_x509_write_extensions( unsigned char **p, unsigned char *start, - mbedtls_asn1_named_data *first ); -int mbedtls_x509_write_names( unsigned char **p, unsigned char *start, - mbedtls_asn1_named_data *first ); -int mbedtls_x509_write_sig( unsigned char **p, unsigned char *start, - const char *oid, size_t oid_len, - unsigned char *sig, size_t size ); - -#define MBEDTLS_X509_SAFE_SNPRINTF \ - do { \ - if( ret < 0 || (size_t) ret >= n ) \ - return( MBEDTLS_ERR_X509_BUFFER_TOO_SMALL ); \ - \ - n -= (size_t) ret; \ - p += (size_t) ret; \ - } while( 0 ) - -#ifdef __cplusplus -} -#endif - -#endif /* x509.h */ diff --git a/windows/mbedtls/x509_crl.h b/windows/mbedtls/x509_crl.h deleted file mode 100644 index 9331827b..00000000 --- a/windows/mbedtls/x509_crl.h +++ /dev/null @@ -1,171 +0,0 @@ -/** - * \file x509_crl.h - * - * \brief X.509 certificate revocation list parsing - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_X509_CRL_H -#define MBEDTLS_X509_CRL_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/x509.h" - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \addtogroup x509_module - * \{ */ - -/** - * \name Structures and functions for parsing CRLs - * \{ - */ - -/** - * Certificate revocation list entry. - * Contains the CA-specific serial numbers and revocation dates. - */ -typedef struct mbedtls_x509_crl_entry -{ - mbedtls_x509_buf MBEDTLS_PRIVATE(raw); - - mbedtls_x509_buf MBEDTLS_PRIVATE(serial); - - mbedtls_x509_time MBEDTLS_PRIVATE(revocation_date); - - mbedtls_x509_buf MBEDTLS_PRIVATE(entry_ext); - - struct mbedtls_x509_crl_entry *MBEDTLS_PRIVATE(next); -} -mbedtls_x509_crl_entry; - -/** - * Certificate revocation list structure. - * Every CRL may have multiple entries. - */ -typedef struct mbedtls_x509_crl -{ - mbedtls_x509_buf MBEDTLS_PRIVATE(raw); /**< The raw certificate data (DER). */ - mbedtls_x509_buf MBEDTLS_PRIVATE(tbs); /**< The raw certificate body (DER). The part that is To Be Signed. */ - - int MBEDTLS_PRIVATE(version); /**< CRL version (1=v1, 2=v2) */ - mbedtls_x509_buf MBEDTLS_PRIVATE(sig_oid); /**< CRL signature type identifier */ - - mbedtls_x509_buf MBEDTLS_PRIVATE(issuer_raw); /**< The raw issuer data (DER). */ - - mbedtls_x509_name MBEDTLS_PRIVATE(issuer); /**< The parsed issuer data (named information object). */ - - mbedtls_x509_time MBEDTLS_PRIVATE(this_update); - mbedtls_x509_time MBEDTLS_PRIVATE(next_update); - - mbedtls_x509_crl_entry MBEDTLS_PRIVATE(entry); /**< The CRL entries containing the certificate revocation times for this CA. */ - - mbedtls_x509_buf MBEDTLS_PRIVATE(crl_ext); - - mbedtls_x509_buf MBEDTLS_PRIVATE(sig_oid2); - mbedtls_x509_buf MBEDTLS_PRIVATE(sig); - mbedtls_md_type_t MBEDTLS_PRIVATE(sig_md); /**< Internal representation of the MD algorithm of the signature algorithm, e.g. MBEDTLS_MD_SHA256 */ - mbedtls_pk_type_t MBEDTLS_PRIVATE(sig_pk); /**< Internal representation of the Public Key algorithm of the signature algorithm, e.g. MBEDTLS_PK_RSA */ - void *MBEDTLS_PRIVATE(sig_opts); /**< Signature options to be passed to mbedtls_pk_verify_ext(), e.g. for RSASSA-PSS */ - - struct mbedtls_x509_crl *MBEDTLS_PRIVATE(next); -} -mbedtls_x509_crl; - -/** - * \brief Parse a DER-encoded CRL and append it to the chained list - * - * \param chain points to the start of the chain - * \param buf buffer holding the CRL data in DER format - * \param buflen size of the buffer - * (including the terminating null byte for PEM data) - * - * \return 0 if successful, or a specific X509 or PEM error code - */ -int mbedtls_x509_crl_parse_der( mbedtls_x509_crl *chain, - const unsigned char *buf, size_t buflen ); -/** - * \brief Parse one or more CRLs and append them to the chained list - * - * \note Multiple CRLs are accepted only if using PEM format - * - * \param chain points to the start of the chain - * \param buf buffer holding the CRL data in PEM or DER format - * \param buflen size of the buffer - * (including the terminating null byte for PEM data) - * - * \return 0 if successful, or a specific X509 or PEM error code - */ -int mbedtls_x509_crl_parse( mbedtls_x509_crl *chain, const unsigned char *buf, size_t buflen ); - -#if defined(MBEDTLS_FS_IO) -/** - * \brief Load one or more CRLs and append them to the chained list - * - * \note Multiple CRLs are accepted only if using PEM format - * - * \param chain points to the start of the chain - * \param path filename to read the CRLs from (in PEM or DER encoding) - * - * \return 0 if successful, or a specific X509 or PEM error code - */ -int mbedtls_x509_crl_parse_file( mbedtls_x509_crl *chain, const char *path ); -#endif /* MBEDTLS_FS_IO */ - -#if !defined(MBEDTLS_X509_REMOVE_INFO) -/** - * \brief Returns an informational string about the CRL. - * - * \param buf Buffer to write to - * \param size Maximum size of buffer - * \param prefix A line prefix - * \param crl The X509 CRL to represent - * - * \return The length of the string written (not including the - * terminated nul byte), or a negative error code. - */ -int mbedtls_x509_crl_info( char *buf, size_t size, const char *prefix, - const mbedtls_x509_crl *crl ); -#endif /* !MBEDTLS_X509_REMOVE_INFO */ - -/** - * \brief Initialize a CRL (chain) - * - * \param crl CRL chain to initialize - */ -void mbedtls_x509_crl_init( mbedtls_x509_crl *crl ); - -/** - * \brief Unallocate all CRL data - * - * \param crl CRL chain to free - */ -void mbedtls_x509_crl_free( mbedtls_x509_crl *crl ); - -/* \} name */ -/* \} addtogroup x509_module */ - -#ifdef __cplusplus -} -#endif - -#endif /* mbedtls_x509_crl.h */ diff --git a/windows/mbedtls/x509_crt.h b/windows/mbedtls/x509_crt.h deleted file mode 100644 index 49211a94..00000000 --- a/windows/mbedtls/x509_crt.h +++ /dev/null @@ -1,1179 +0,0 @@ -/** - * \file x509_crt.h - * - * \brief X.509 certificate parsing and writing - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_X509_CRT_H -#define MBEDTLS_X509_CRT_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/x509.h" -#include "mbedtls/x509_crl.h" -#include "mbedtls/bignum.h" - -/** - * \addtogroup x509_module - * \{ - */ - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \name Structures and functions for parsing and writing X.509 certificates - * \{ - */ - -/** - * Container for an X.509 certificate. The certificate may be chained. - */ -typedef struct mbedtls_x509_crt -{ - int MBEDTLS_PRIVATE(own_buffer); /**< Indicates if \c raw is owned - * by the structure or not. */ - mbedtls_x509_buf MBEDTLS_PRIVATE(raw); /**< The raw certificate data (DER). */ - mbedtls_x509_buf MBEDTLS_PRIVATE(tbs); /**< The raw certificate body (DER). The part that is To Be Signed. */ - - int MBEDTLS_PRIVATE(version); /**< The X.509 version. (1=v1, 2=v2, 3=v3) */ - mbedtls_x509_buf MBEDTLS_PRIVATE(serial); /**< Unique id for certificate issued by a specific CA. */ - mbedtls_x509_buf MBEDTLS_PRIVATE(sig_oid); /**< Signature algorithm, e.g. sha1RSA */ - - mbedtls_x509_buf MBEDTLS_PRIVATE(issuer_raw); /**< The raw issuer data (DER). Used for quick comparison. */ - mbedtls_x509_buf MBEDTLS_PRIVATE(subject_raw); /**< The raw subject data (DER). Used for quick comparison. */ - - mbedtls_x509_name MBEDTLS_PRIVATE(issuer); /**< The parsed issuer data (named information object). */ - mbedtls_x509_name MBEDTLS_PRIVATE(subject); /**< The parsed subject data (named information object). */ - - mbedtls_x509_time MBEDTLS_PRIVATE(valid_from); /**< Start time of certificate validity. */ - mbedtls_x509_time MBEDTLS_PRIVATE(valid_to); /**< End time of certificate validity. */ - - mbedtls_x509_buf MBEDTLS_PRIVATE(pk_raw); - mbedtls_pk_context MBEDTLS_PRIVATE(pk); /**< Container for the public key context. */ - - mbedtls_x509_buf MBEDTLS_PRIVATE(issuer_id); /**< Optional X.509 v2/v3 issuer unique identifier. */ - mbedtls_x509_buf MBEDTLS_PRIVATE(subject_id); /**< Optional X.509 v2/v3 subject unique identifier. */ - mbedtls_x509_buf MBEDTLS_PRIVATE(v3_ext); /**< Optional X.509 v3 extensions. */ - mbedtls_x509_sequence MBEDTLS_PRIVATE(subject_alt_names); /**< Optional list of raw entries of Subject Alternative Names extension (currently only dNSName and OtherName are listed). */ - - mbedtls_x509_sequence MBEDTLS_PRIVATE(certificate_policies); /**< Optional list of certificate policies (Only anyPolicy is printed and enforced, however the rest of the policies are still listed). */ - - int MBEDTLS_PRIVATE(ext_types); /**< Bit string containing detected and parsed extensions */ - int MBEDTLS_PRIVATE(ca_istrue); /**< Optional Basic Constraint extension value: 1 if this certificate belongs to a CA, 0 otherwise. */ - int MBEDTLS_PRIVATE(max_pathlen); /**< Optional Basic Constraint extension value: The maximum path length to the root certificate. Path length is 1 higher than RFC 5280 'meaning', so 1+ */ - - unsigned int MBEDTLS_PRIVATE(key_usage); /**< Optional key usage extension value: See the values in x509.h */ - - mbedtls_x509_sequence MBEDTLS_PRIVATE(ext_key_usage); /**< Optional list of extended key usage OIDs. */ - - unsigned char MBEDTLS_PRIVATE(ns_cert_type); /**< Optional Netscape certificate type extension value: See the values in x509.h */ - - mbedtls_x509_buf MBEDTLS_PRIVATE(sig); /**< Signature: hash of the tbs part signed with the private key. */ - mbedtls_md_type_t MBEDTLS_PRIVATE(sig_md); /**< Internal representation of the MD algorithm of the signature algorithm, e.g. MBEDTLS_MD_SHA256 */ - mbedtls_pk_type_t MBEDTLS_PRIVATE(sig_pk); /**< Internal representation of the Public Key algorithm of the signature algorithm, e.g. MBEDTLS_PK_RSA */ - void *MBEDTLS_PRIVATE(sig_opts); /**< Signature options to be passed to mbedtls_pk_verify_ext(), e.g. for RSASSA-PSS */ - - struct mbedtls_x509_crt *MBEDTLS_PRIVATE(next); /**< Next certificate in the CA-chain. */ -} -mbedtls_x509_crt; - -/** - * From RFC 5280 section 4.2.1.6: - * OtherName ::= SEQUENCE { - * type-id OBJECT IDENTIFIER, - * value [0] EXPLICIT ANY DEFINED BY type-id } - */ -typedef struct mbedtls_x509_san_other_name -{ - /** - * The type_id is an OID as deifned in RFC 5280. - * To check the value of the type id, you should use - * \p MBEDTLS_OID_CMP with a known OID mbedtls_x509_buf. - */ - mbedtls_x509_buf MBEDTLS_PRIVATE(type_id); /**< The type id. */ - union - { - /** - * From RFC 4108 section 5: - * HardwareModuleName ::= SEQUENCE { - * hwType OBJECT IDENTIFIER, - * hwSerialNum OCTET STRING } - */ - struct - { - mbedtls_x509_buf MBEDTLS_PRIVATE(oid); /**< The object identifier. */ - mbedtls_x509_buf MBEDTLS_PRIVATE(val); /**< The named value. */ - } - MBEDTLS_PRIVATE(hardware_module_name); - } - MBEDTLS_PRIVATE(value); -} -mbedtls_x509_san_other_name; - -/** - * A structure for holding the parsed Subject Alternative Name, according to type - */ -typedef struct mbedtls_x509_subject_alternative_name -{ - int MBEDTLS_PRIVATE(type); /**< The SAN type, value of MBEDTLS_X509_SAN_XXX. */ - union { - mbedtls_x509_san_other_name MBEDTLS_PRIVATE(other_name); /**< The otherName supported type. */ - mbedtls_x509_buf MBEDTLS_PRIVATE(unstructured_name); /**< The buffer for the un constructed types. Only dnsName currently supported */ - } - MBEDTLS_PRIVATE(san); /**< A union of the supported SAN types */ -} -mbedtls_x509_subject_alternative_name; - -/** - * Build flag from an algorithm/curve identifier (pk, md, ecp) - * Since 0 is always XXX_NONE, ignore it. - */ -#define MBEDTLS_X509_ID_FLAG( id ) ( 1 << ( (id) - 1 ) ) - -/** - * Security profile for certificate verification. - * - * All lists are bitfields, built by ORing flags from MBEDTLS_X509_ID_FLAG(). - * - * The fields of this structure are part of the public API and can be - * manipulated directly by applications. Future versions of the library may - * add extra fields or reorder existing fields. - * - * You can create custom profiles by starting from a copy of - * an existing profile, such as mbedtls_x509_crt_profile_default or - * mbedtls_x509_ctr_profile_none and then tune it to your needs. - * - * For example to allow SHA-224 in addition to the default: - * - * mbedtls_x509_crt_profile my_profile = mbedtls_x509_crt_profile_default; - * my_profile.allowed_mds |= MBEDTLS_X509_ID_FLAG( MBEDTLS_MD_SHA224 ); - * - * Or to allow only RSA-3072+ with SHA-256: - * - * mbedtls_x509_crt_profile my_profile = mbedtls_x509_crt_profile_none; - * my_profile.allowed_mds = MBEDTLS_X509_ID_FLAG( MBEDTLS_MD_SHA256 ); - * my_profile.allowed_pks = MBEDTLS_X509_ID_FLAG( MBEDTLS_PK_RSA ); - * my_profile.rsa_min_bitlen = 3072; - */ -typedef struct mbedtls_x509_crt_profile -{ - uint32_t allowed_mds; /**< MDs for signatures */ - uint32_t allowed_pks; /**< PK algs for signatures */ - uint32_t allowed_curves; /**< Elliptic curves for ECDSA */ - uint32_t rsa_min_bitlen; /**< Minimum size for RSA keys */ -} -mbedtls_x509_crt_profile; - -#define MBEDTLS_X509_CRT_VERSION_1 0 -#define MBEDTLS_X509_CRT_VERSION_2 1 -#define MBEDTLS_X509_CRT_VERSION_3 2 - -#define MBEDTLS_X509_RFC5280_MAX_SERIAL_LEN 32 -#define MBEDTLS_X509_RFC5280_UTC_TIME_LEN 15 - -#if !defined( MBEDTLS_X509_MAX_FILE_PATH_LEN ) -#define MBEDTLS_X509_MAX_FILE_PATH_LEN 512 -#endif - -/* This macro unfolds to the concatenation of macro invocations - * X509_CRT_ERROR_INFO( error code, - * error code as string, - * human readable description ) - * where X509_CRT_ERROR_INFO is defined by the user. - * See x509_crt.c for an example of how to use this. */ -#define MBEDTLS_X509_CRT_ERROR_INFO_LIST \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_EXPIRED, \ - "MBEDTLS_X509_BADCERT_EXPIRED", \ - "The certificate validity has expired" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_REVOKED, \ - "MBEDTLS_X509_BADCERT_REVOKED", \ - "The certificate has been revoked (is on a CRL)" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_CN_MISMATCH, \ - "MBEDTLS_X509_BADCERT_CN_MISMATCH", \ - "The certificate Common Name (CN) does not match with the expected CN" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_NOT_TRUSTED, \ - "MBEDTLS_X509_BADCERT_NOT_TRUSTED", \ - "The certificate is not correctly signed by the trusted CA" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCRL_NOT_TRUSTED, \ - "MBEDTLS_X509_BADCRL_NOT_TRUSTED", \ - "The CRL is not correctly signed by the trusted CA" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCRL_EXPIRED, \ - "MBEDTLS_X509_BADCRL_EXPIRED", \ - "The CRL is expired" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_MISSING, \ - "MBEDTLS_X509_BADCERT_MISSING", \ - "Certificate was missing" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_SKIP_VERIFY, \ - "MBEDTLS_X509_BADCERT_SKIP_VERIFY", \ - "Certificate verification was skipped" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_OTHER, \ - "MBEDTLS_X509_BADCERT_OTHER", \ - "Other reason (can be used by verify callback)" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_FUTURE, \ - "MBEDTLS_X509_BADCERT_FUTURE", \ - "The certificate validity starts in the future" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCRL_FUTURE, \ - "MBEDTLS_X509_BADCRL_FUTURE", \ - "The CRL is from the future" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_KEY_USAGE, \ - "MBEDTLS_X509_BADCERT_KEY_USAGE", \ - "Usage does not match the keyUsage extension" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_EXT_KEY_USAGE, \ - "MBEDTLS_X509_BADCERT_EXT_KEY_USAGE", \ - "Usage does not match the extendedKeyUsage extension" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_NS_CERT_TYPE, \ - "MBEDTLS_X509_BADCERT_NS_CERT_TYPE", \ - "Usage does not match the nsCertType extension" ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_BAD_MD, \ - "MBEDTLS_X509_BADCERT_BAD_MD", \ - "The certificate is signed with an unacceptable hash." ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_BAD_PK, \ - "MBEDTLS_X509_BADCERT_BAD_PK", \ - "The certificate is signed with an unacceptable PK alg (eg RSA vs ECDSA)." ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCERT_BAD_KEY, \ - "MBEDTLS_X509_BADCERT_BAD_KEY", \ - "The certificate is signed with an unacceptable key (eg bad curve, RSA too short)." ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCRL_BAD_MD, \ - "MBEDTLS_X509_BADCRL_BAD_MD", \ - "The CRL is signed with an unacceptable hash." ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCRL_BAD_PK, \ - "MBEDTLS_X509_BADCRL_BAD_PK", \ - "The CRL is signed with an unacceptable PK alg (eg RSA vs ECDSA)." ) \ - X509_CRT_ERROR_INFO( MBEDTLS_X509_BADCRL_BAD_KEY, \ - "MBEDTLS_X509_BADCRL_BAD_KEY", \ - "The CRL is signed with an unacceptable key (eg bad curve, RSA too short)." ) - -/** - * Container for writing a certificate (CRT) - */ -typedef struct mbedtls_x509write_cert -{ - int MBEDTLS_PRIVATE(version); - mbedtls_mpi MBEDTLS_PRIVATE(serial); - mbedtls_pk_context *MBEDTLS_PRIVATE(subject_key); - mbedtls_pk_context *MBEDTLS_PRIVATE(issuer_key); - mbedtls_asn1_named_data *MBEDTLS_PRIVATE(subject); - mbedtls_asn1_named_data *MBEDTLS_PRIVATE(issuer); - mbedtls_md_type_t MBEDTLS_PRIVATE(md_alg); - char MBEDTLS_PRIVATE(not_before)[MBEDTLS_X509_RFC5280_UTC_TIME_LEN + 1]; - char MBEDTLS_PRIVATE(not_after)[MBEDTLS_X509_RFC5280_UTC_TIME_LEN + 1]; - mbedtls_asn1_named_data *MBEDTLS_PRIVATE(extensions); -} -mbedtls_x509write_cert; - -/** - * Item in a verification chain: cert and flags for it - */ -typedef struct { - mbedtls_x509_crt *MBEDTLS_PRIVATE(crt); - uint32_t MBEDTLS_PRIVATE(flags); -} mbedtls_x509_crt_verify_chain_item; - -/** - * Max size of verification chain: end-entity + intermediates + trusted root - */ -#define MBEDTLS_X509_MAX_VERIFY_CHAIN_SIZE ( MBEDTLS_X509_MAX_INTERMEDIATE_CA + 2 ) - -/** - * Verification chain as built by \c mbedtls_crt_verify_chain() - */ -typedef struct -{ - mbedtls_x509_crt_verify_chain_item MBEDTLS_PRIVATE(items)[MBEDTLS_X509_MAX_VERIFY_CHAIN_SIZE]; - unsigned MBEDTLS_PRIVATE(len); - -#if defined(MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK) - /* This stores the list of potential trusted signers obtained from - * the CA callback used for the CRT verification, if configured. - * We must track it somewhere because the callback passes its - * ownership to the caller. */ - mbedtls_x509_crt *MBEDTLS_PRIVATE(trust_ca_cb_result); -#endif /* MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK */ -} mbedtls_x509_crt_verify_chain; - -#if defined(MBEDTLS_ECDSA_C) && defined(MBEDTLS_ECP_RESTARTABLE) - -/** - * \brief Context for resuming X.509 verify operations - */ -typedef struct -{ - /* for check_signature() */ - mbedtls_pk_restart_ctx MBEDTLS_PRIVATE(pk); - - /* for find_parent_in() */ - mbedtls_x509_crt *MBEDTLS_PRIVATE(parent); /* non-null iff parent_in in progress */ - mbedtls_x509_crt *MBEDTLS_PRIVATE(fallback_parent); - int MBEDTLS_PRIVATE(fallback_signature_is_good); - - /* for find_parent() */ - int MBEDTLS_PRIVATE(parent_is_trusted); /* -1 if find_parent is not in progress */ - - /* for verify_chain() */ - enum { - x509_crt_rs_none, - x509_crt_rs_find_parent, - } MBEDTLS_PRIVATE(in_progress); /* none if no operation is in progress */ - int MBEDTLS_PRIVATE(self_cnt); - mbedtls_x509_crt_verify_chain MBEDTLS_PRIVATE(ver_chain); - -} mbedtls_x509_crt_restart_ctx; - -#else /* MBEDTLS_ECDSA_C && MBEDTLS_ECP_RESTARTABLE */ - -/* Now we can declare functions that take a pointer to that */ -typedef void mbedtls_x509_crt_restart_ctx; - -#endif /* MBEDTLS_ECDSA_C && MBEDTLS_ECP_RESTARTABLE */ - -#if defined(MBEDTLS_X509_CRT_PARSE_C) -/** - * Default security profile. Should provide a good balance between security - * and compatibility with current deployments. - * - * This profile permits: - * - SHA2 hashes with at least 256 bits: SHA-256, SHA-384, SHA-512. - * - Elliptic curves with 255 bits and above except secp256k1. - * - RSA with 2048 bits and above. - * - * New minor versions of Mbed TLS may extend this profile, for example if - * new algorithms are added to the library. New minor versions of Mbed TLS will - * not reduce this profile unless serious security concerns require it. - */ -extern const mbedtls_x509_crt_profile mbedtls_x509_crt_profile_default; - -/** - * Expected next default profile. Recommended for new deployments. - * Currently targets a 128-bit security level, except for allowing RSA-2048. - * This profile may change at any time. - */ -extern const mbedtls_x509_crt_profile mbedtls_x509_crt_profile_next; - -/** - * NSA Suite B profile. - */ -extern const mbedtls_x509_crt_profile mbedtls_x509_crt_profile_suiteb; - -/** - * Empty profile that allows nothing. Useful as a basis for constructing - * custom profiles. - */ -extern const mbedtls_x509_crt_profile mbedtls_x509_crt_profile_none; - -/** - * \brief Parse a single DER formatted certificate and add it - * to the end of the provided chained list. - * - * \param chain The pointer to the start of the CRT chain to attach to. - * When parsing the first CRT in a chain, this should point - * to an instance of ::mbedtls_x509_crt initialized through - * mbedtls_x509_crt_init(). - * \param buf The buffer holding the DER encoded certificate. - * \param buflen The size in Bytes of \p buf. - * - * \note This function makes an internal copy of the CRT buffer - * \p buf. In particular, \p buf may be destroyed or reused - * after this call returns. To avoid duplicating the CRT - * buffer (at the cost of stricter lifetime constraints), - * use mbedtls_x509_crt_parse_der_nocopy() instead. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_x509_crt_parse_der( mbedtls_x509_crt *chain, - const unsigned char *buf, - size_t buflen ); - -/** - * \brief The type of certificate extension callbacks. - * - * Callbacks of this type are passed to and used by the - * mbedtls_x509_crt_parse_der_with_ext_cb() routine when - * it encounters either an unsupported extension or a - * "certificate policies" extension containing any - * unsupported certificate policies. - * Future versions of the library may invoke the callback - * in other cases, if and when the need arises. - * - * \param p_ctx An opaque context passed to the callback. - * \param crt The certificate being parsed. - * \param oid The OID of the extension. - * \param critical Whether the extension is critical. - * \param p Pointer to the start of the extension value - * (the content of the OCTET STRING). - * \param end End of extension value. - * - * \note The callback must fail and return a negative error code - * if it can not parse or does not support the extension. - * When the callback fails to parse a critical extension - * mbedtls_x509_crt_parse_der_with_ext_cb() also fails. - * When the callback fails to parse a non critical extension - * mbedtls_x509_crt_parse_der_with_ext_cb() simply skips - * the extension and continues parsing. - * - * \return \c 0 on success. - * \return A negative error code on failure. - */ -typedef int (*mbedtls_x509_crt_ext_cb_t)( void *p_ctx, - mbedtls_x509_crt const *crt, - mbedtls_x509_buf const *oid, - int critical, - const unsigned char *p, - const unsigned char *end ); - -/** - * \brief Parse a single DER formatted certificate and add it - * to the end of the provided chained list. - * - * \param chain The pointer to the start of the CRT chain to attach to. - * When parsing the first CRT in a chain, this should point - * to an instance of ::mbedtls_x509_crt initialized through - * mbedtls_x509_crt_init(). - * \param buf The buffer holding the DER encoded certificate. - * \param buflen The size in Bytes of \p buf. - * \param make_copy When not zero this function makes an internal copy of the - * CRT buffer \p buf. In particular, \p buf may be destroyed - * or reused after this call returns. - * When zero this function avoids duplicating the CRT buffer - * by taking temporary ownership thereof until the CRT - * is destroyed (like mbedtls_x509_crt_parse_der_nocopy()) - * \param cb A callback invoked for every unsupported certificate - * extension. - * \param p_ctx An opaque context passed to the callback. - * - * \note This call is functionally equivalent to - * mbedtls_x509_crt_parse_der(), and/or - * mbedtls_x509_crt_parse_der_nocopy() - * but it calls the callback with every unsupported - * certificate extension and additionally the - * "certificate policies" extension if it contains any - * unsupported certificate policies. - * The callback must return a negative error code if it - * does not know how to handle such an extension. - * When the callback fails to parse a critical extension - * mbedtls_x509_crt_parse_der_with_ext_cb() also fails. - * When the callback fails to parse a non critical extension - * mbedtls_x509_crt_parse_der_with_ext_cb() simply skips - * the extension and continues parsing. - * Future versions of the library may invoke the callback - * in other cases, if and when the need arises. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_x509_crt_parse_der_with_ext_cb( mbedtls_x509_crt *chain, - const unsigned char *buf, - size_t buflen, - int make_copy, - mbedtls_x509_crt_ext_cb_t cb, - void *p_ctx ); - -/** - * \brief Parse a single DER formatted certificate and add it - * to the end of the provided chained list. This is a - * variant of mbedtls_x509_crt_parse_der() which takes - * temporary ownership of the CRT buffer until the CRT - * is destroyed. - * - * \param chain The pointer to the start of the CRT chain to attach to. - * When parsing the first CRT in a chain, this should point - * to an instance of ::mbedtls_x509_crt initialized through - * mbedtls_x509_crt_init(). - * \param buf The address of the readable buffer holding the DER encoded - * certificate to use. On success, this buffer must be - * retained and not be changed for the liftetime of the - * CRT chain \p chain, that is, until \p chain is destroyed - * through a call to mbedtls_x509_crt_free(). - * \param buflen The size in Bytes of \p buf. - * - * \note This call is functionally equivalent to - * mbedtls_x509_crt_parse_der(), but it avoids creating a - * copy of the input buffer at the cost of stronger lifetime - * constraints. This is useful in constrained environments - * where duplication of the CRT cannot be tolerated. - * - * \return \c 0 if successful. - * \return A negative error code on failure. - */ -int mbedtls_x509_crt_parse_der_nocopy( mbedtls_x509_crt *chain, - const unsigned char *buf, - size_t buflen ); - -/** - * \brief Parse one DER-encoded or one or more concatenated PEM-encoded - * certificates and add them to the chained list. - * - * For CRTs in PEM encoding, the function parses permissively: - * if at least one certificate can be parsed, the function - * returns the number of certificates for which parsing failed - * (hence \c 0 if all certificates were parsed successfully). - * If no certificate could be parsed, the function returns - * the first (negative) error encountered during parsing. - * - * PEM encoded certificates may be interleaved by other data - * such as human readable descriptions of their content, as - * long as the certificates are enclosed in the PEM specific - * '-----{BEGIN/END} CERTIFICATE-----' delimiters. - * - * \param chain The chain to which to add the parsed certificates. - * \param buf The buffer holding the certificate data in PEM or DER format. - * For certificates in PEM encoding, this may be a concatenation - * of multiple certificates; for DER encoding, the buffer must - * comprise exactly one certificate. - * \param buflen The size of \p buf, including the terminating \c NULL byte - * in case of PEM encoded data. - * - * \return \c 0 if all certificates were parsed successfully. - * \return The (positive) number of certificates that couldn't - * be parsed if parsing was partly successful (see above). - * \return A negative X509 or PEM error code otherwise. - * - */ -int mbedtls_x509_crt_parse( mbedtls_x509_crt *chain, const unsigned char *buf, size_t buflen ); - -#if defined(MBEDTLS_FS_IO) -/** - * \brief Load one or more certificates and add them - * to the chained list. Parses permissively. If some - * certificates can be parsed, the result is the number - * of failed certificates it encountered. If none complete - * correctly, the first error is returned. - * - * \param chain points to the start of the chain - * \param path filename to read the certificates from - * - * \return 0 if all certificates parsed successfully, a positive number - * if partly successful or a specific X509 or PEM error code - */ -int mbedtls_x509_crt_parse_file( mbedtls_x509_crt *chain, const char *path ); - -/** - * \brief Load one or more certificate files from a path and add them - * to the chained list. Parses permissively. If some - * certificates can be parsed, the result is the number - * of failed certificates it encountered. If none complete - * correctly, the first error is returned. - * - * \param chain points to the start of the chain - * \param path directory / folder to read the certificate files from - * - * \return 0 if all certificates parsed successfully, a positive number - * if partly successful or a specific X509 or PEM error code - */ -int mbedtls_x509_crt_parse_path( mbedtls_x509_crt *chain, const char *path ); - -#endif /* MBEDTLS_FS_IO */ -/** - * \brief This function parses an item in the SubjectAlternativeNames - * extension. - * - * \param san_buf The buffer holding the raw data item of the subject - * alternative name. - * \param san The target structure to populate with the parsed presentation - * of the subject alternative name encoded in \p san_raw. - * - * \note Only "dnsName" and "otherName" of type hardware_module_name - * as defined in RFC 4180 is supported. - * - * \note This function should be called on a single raw data of - * subject alternative name. For example, after successful - * certificate parsing, one must iterate on every item in the - * \p crt->subject_alt_names sequence, and pass it to - * this function. - * - * \warning The target structure contains pointers to the raw data of the - * parsed certificate, and its lifetime is restricted by the - * lifetime of the certificate. - * - * \return \c 0 on success - * \return #MBEDTLS_ERR_X509_FEATURE_UNAVAILABLE for an unsupported - * SAN type. - * \return Another negative value for any other failure. - */ -int mbedtls_x509_parse_subject_alt_name( const mbedtls_x509_buf *san_buf, - mbedtls_x509_subject_alternative_name *san ); - -#if !defined(MBEDTLS_X509_REMOVE_INFO) -/** - * \brief Returns an informational string about the - * certificate. - * - * \param buf Buffer to write to - * \param size Maximum size of buffer - * \param prefix A line prefix - * \param crt The X509 certificate to represent - * - * \return The length of the string written (not including the - * terminated nul byte), or a negative error code. - */ -int mbedtls_x509_crt_info( char *buf, size_t size, const char *prefix, - const mbedtls_x509_crt *crt ); - -/** - * \brief Returns an informational string about the - * verification status of a certificate. - * - * \param buf Buffer to write to - * \param size Maximum size of buffer - * \param prefix A line prefix - * \param flags Verification flags created by mbedtls_x509_crt_verify() - * - * \return The length of the string written (not including the - * terminated nul byte), or a negative error code. - */ -int mbedtls_x509_crt_verify_info( char *buf, size_t size, const char *prefix, - uint32_t flags ); -#endif /* !MBEDTLS_X509_REMOVE_INFO */ - -/** - * \brief Verify a chain of certificates. - * - * The verify callback is a user-supplied callback that - * can clear / modify / add flags for a certificate. If set, - * the verification callback is called for each - * certificate in the chain (from the trust-ca down to the - * presented crt). The parameters for the callback are: - * (void *parameter, mbedtls_x509_crt *crt, int certificate_depth, - * int *flags). With the flags representing current flags for - * that specific certificate and the certificate depth from - * the bottom (Peer cert depth = 0). - * - * All flags left after returning from the callback - * are also returned to the application. The function should - * return 0 for anything (including invalid certificates) - * other than fatal error, as a non-zero return code - * immediately aborts the verification process. For fatal - * errors, a specific error code should be used (different - * from MBEDTLS_ERR_X509_CERT_VERIFY_FAILED which should not - * be returned at this point), or MBEDTLS_ERR_X509_FATAL_ERROR - * can be used if no better code is available. - * - * \note In case verification failed, the results can be displayed - * using \c mbedtls_x509_crt_verify_info() - * - * \note Same as \c mbedtls_x509_crt_verify_with_profile() with the - * default security profile. - * - * \note It is your responsibility to provide up-to-date CRLs for - * all trusted CAs. If no CRL is provided for the CA that was - * used to sign the certificate, CRL verification is skipped - * silently, that is *without* setting any flag. - * - * \note The \c trust_ca list can contain two types of certificates: - * (1) those of trusted root CAs, so that certificates - * chaining up to those CAs will be trusted, and (2) - * self-signed end-entity certificates to be trusted (for - * specific peers you know) - in that case, the self-signed - * certificate doesn't need to have the CA bit set. - * - * \param crt The certificate chain to be verified. - * \param trust_ca The list of trusted CAs. - * \param ca_crl The list of CRLs for trusted CAs. - * \param cn The expected Common Name. This will be checked to be - * present in the certificate's subjectAltNames extension or, - * if this extension is absent, as a CN component in its - * Subject name. Currently only DNS names are supported. This - * may be \c NULL if the CN need not be verified. - * \param flags The address at which to store the result of the verification. - * If the verification couldn't be completed, the flag value is - * set to (uint32_t) -1. - * \param f_vrfy The verification callback to use. See the documentation - * of mbedtls_x509_crt_verify() for more information. - * \param p_vrfy The context to be passed to \p f_vrfy. - * - * \return \c 0 if the chain is valid with respect to the - * passed CN, CAs, CRLs and security profile. - * \return #MBEDTLS_ERR_X509_CERT_VERIFY_FAILED in case the - * certificate chain verification failed. In this case, - * \c *flags will have one or more - * \c MBEDTLS_X509_BADCERT_XXX or \c MBEDTLS_X509_BADCRL_XXX - * flags set. - * \return Another negative error code in case of a fatal error - * encountered during the verification process. - */ -int mbedtls_x509_crt_verify( mbedtls_x509_crt *crt, - mbedtls_x509_crt *trust_ca, - mbedtls_x509_crl *ca_crl, - const char *cn, uint32_t *flags, - int (*f_vrfy)(void *, mbedtls_x509_crt *, int, uint32_t *), - void *p_vrfy ); - -/** - * \brief Verify a chain of certificates with respect to - * a configurable security profile. - * - * \note Same as \c mbedtls_x509_crt_verify(), but with explicit - * security profile. - * - * \note The restrictions on keys (RSA minimum size, allowed curves - * for ECDSA) apply to all certificates: trusted root, - * intermediate CAs if any, and end entity certificate. - * - * \param crt The certificate chain to be verified. - * \param trust_ca The list of trusted CAs. - * \param ca_crl The list of CRLs for trusted CAs. - * \param profile The security profile to use for the verification. - * \param cn The expected Common Name. This may be \c NULL if the - * CN need not be verified. - * \param flags The address at which to store the result of the verification. - * If the verification couldn't be completed, the flag value is - * set to (uint32_t) -1. - * \param f_vrfy The verification callback to use. See the documentation - * of mbedtls_x509_crt_verify() for more information. - * \param p_vrfy The context to be passed to \p f_vrfy. - * - * \return \c 0 if the chain is valid with respect to the - * passed CN, CAs, CRLs and security profile. - * \return #MBEDTLS_ERR_X509_CERT_VERIFY_FAILED in case the - * certificate chain verification failed. In this case, - * \c *flags will have one or more - * \c MBEDTLS_X509_BADCERT_XXX or \c MBEDTLS_X509_BADCRL_XXX - * flags set. - * \return Another negative error code in case of a fatal error - * encountered during the verification process. - */ -int mbedtls_x509_crt_verify_with_profile( mbedtls_x509_crt *crt, - mbedtls_x509_crt *trust_ca, - mbedtls_x509_crl *ca_crl, - const mbedtls_x509_crt_profile *profile, - const char *cn, uint32_t *flags, - int (*f_vrfy)(void *, mbedtls_x509_crt *, int, uint32_t *), - void *p_vrfy ); - -/** - * \brief Restartable version of \c mbedtls_crt_verify_with_profile() - * - * \note Performs the same job as \c mbedtls_crt_verify_with_profile() - * but can return early and restart according to the limit - * set with \c mbedtls_ecp_set_max_ops() to reduce blocking. - * - * \param crt The certificate chain to be verified. - * \param trust_ca The list of trusted CAs. - * \param ca_crl The list of CRLs for trusted CAs. - * \param profile The security profile to use for the verification. - * \param cn The expected Common Name. This may be \c NULL if the - * CN need not be verified. - * \param flags The address at which to store the result of the verification. - * If the verification couldn't be completed, the flag value is - * set to (uint32_t) -1. - * \param f_vrfy The verification callback to use. See the documentation - * of mbedtls_x509_crt_verify() for more information. - * \param p_vrfy The context to be passed to \p f_vrfy. - * \param rs_ctx The restart context to use. This may be set to \c NULL - * to disable restartable ECC. - * - * \return See \c mbedtls_crt_verify_with_profile(), or - * \return #MBEDTLS_ERR_ECP_IN_PROGRESS if maximum number of - * operations was reached: see \c mbedtls_ecp_set_max_ops(). - */ -int mbedtls_x509_crt_verify_restartable( mbedtls_x509_crt *crt, - mbedtls_x509_crt *trust_ca, - mbedtls_x509_crl *ca_crl, - const mbedtls_x509_crt_profile *profile, - const char *cn, uint32_t *flags, - int (*f_vrfy)(void *, mbedtls_x509_crt *, int, uint32_t *), - void *p_vrfy, - mbedtls_x509_crt_restart_ctx *rs_ctx ); - -/** - * \brief The type of trusted certificate callbacks. - * - * Callbacks of this type are passed to and used by the CRT - * verification routine mbedtls_x509_crt_verify_with_ca_cb() - * when looking for trusted signers of a given certificate. - * - * On success, the callback returns a list of trusted - * certificates to be considered as potential signers - * for the input certificate. - * - * \param p_ctx An opaque context passed to the callback. - * \param child The certificate for which to search a potential signer. - * This will point to a readable certificate. - * \param candidate_cas The address at which to store the address of the first - * entry in the generated linked list of candidate signers. - * This will not be \c NULL. - * - * \note The callback must only return a non-zero value on a - * fatal error. If, in contrast, the search for a potential - * signer completes without a single candidate, the - * callback must return \c 0 and set \c *candidate_cas - * to \c NULL. - * - * \return \c 0 on success. In this case, \c *candidate_cas points - * to a heap-allocated linked list of instances of - * ::mbedtls_x509_crt, and ownership of this list is passed - * to the caller. - * \return A negative error code on failure. - */ -typedef int (*mbedtls_x509_crt_ca_cb_t)( void *p_ctx, - mbedtls_x509_crt const *child, - mbedtls_x509_crt **candidate_cas ); - -#if defined(MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK) -/** - * \brief Version of \c mbedtls_x509_crt_verify_with_profile() which - * uses a callback to acquire the list of trusted CA - * certificates. - * - * \param crt The certificate chain to be verified. - * \param f_ca_cb The callback to be used to query for potential signers - * of a given child certificate. See the documentation of - * ::mbedtls_x509_crt_ca_cb_t for more information. - * \param p_ca_cb The opaque context to be passed to \p f_ca_cb. - * \param profile The security profile for the verification. - * \param cn The expected Common Name. This may be \c NULL if the - * CN need not be verified. - * \param flags The address at which to store the result of the verification. - * If the verification couldn't be completed, the flag value is - * set to (uint32_t) -1. - * \param f_vrfy The verification callback to use. See the documentation - * of mbedtls_x509_crt_verify() for more information. - * \param p_vrfy The context to be passed to \p f_vrfy. - * - * \return See \c mbedtls_crt_verify_with_profile(). - */ -int mbedtls_x509_crt_verify_with_ca_cb( mbedtls_x509_crt *crt, - mbedtls_x509_crt_ca_cb_t f_ca_cb, - void *p_ca_cb, - const mbedtls_x509_crt_profile *profile, - const char *cn, uint32_t *flags, - int (*f_vrfy)(void *, mbedtls_x509_crt *, int, uint32_t *), - void *p_vrfy ); - -#endif /* MBEDTLS_X509_TRUSTED_CERTIFICATE_CALLBACK */ - -/** - * \brief Check usage of certificate against keyUsage extension. - * - * \param crt Leaf certificate used. - * \param usage Intended usage(s) (eg MBEDTLS_X509_KU_KEY_ENCIPHERMENT - * before using the certificate to perform an RSA key - * exchange). - * - * \note Except for decipherOnly and encipherOnly, a bit set in the - * usage argument means this bit MUST be set in the - * certificate. For decipherOnly and encipherOnly, it means - * that bit MAY be set. - * - * \return 0 is these uses of the certificate are allowed, - * MBEDTLS_ERR_X509_BAD_INPUT_DATA if the keyUsage extension - * is present but does not match the usage argument. - * - * \note You should only call this function on leaf certificates, on - * (intermediate) CAs the keyUsage extension is automatically - * checked by \c mbedtls_x509_crt_verify(). - */ -int mbedtls_x509_crt_check_key_usage( const mbedtls_x509_crt *crt, - unsigned int usage ); - -/** - * \brief Check usage of certificate against extendedKeyUsage. - * - * \param crt Leaf certificate used. - * \param usage_oid Intended usage (eg MBEDTLS_OID_SERVER_AUTH or - * MBEDTLS_OID_CLIENT_AUTH). - * \param usage_len Length of usage_oid (eg given by MBEDTLS_OID_SIZE()). - * - * \return 0 if this use of the certificate is allowed, - * MBEDTLS_ERR_X509_BAD_INPUT_DATA if not. - * - * \note Usually only makes sense on leaf certificates. - */ -int mbedtls_x509_crt_check_extended_key_usage( const mbedtls_x509_crt *crt, - const char *usage_oid, - size_t usage_len ); - -#if defined(MBEDTLS_X509_CRL_PARSE_C) -/** - * \brief Verify the certificate revocation status - * - * \param crt a certificate to be verified - * \param crl the CRL to verify against - * - * \return 1 if the certificate is revoked, 0 otherwise - * - */ -int mbedtls_x509_crt_is_revoked( const mbedtls_x509_crt *crt, const mbedtls_x509_crl *crl ); -#endif /* MBEDTLS_X509_CRL_PARSE_C */ - -/** - * \brief Initialize a certificate (chain) - * - * \param crt Certificate chain to initialize - */ -void mbedtls_x509_crt_init( mbedtls_x509_crt *crt ); - -/** - * \brief Unallocate all certificate data - * - * \param crt Certificate chain to free - */ -void mbedtls_x509_crt_free( mbedtls_x509_crt *crt ); - -#if defined(MBEDTLS_ECDSA_C) && defined(MBEDTLS_ECP_RESTARTABLE) -/** - * \brief Initialize a restart context - */ -void mbedtls_x509_crt_restart_init( mbedtls_x509_crt_restart_ctx *ctx ); - -/** - * \brief Free the components of a restart context - */ -void mbedtls_x509_crt_restart_free( mbedtls_x509_crt_restart_ctx *ctx ); -#endif /* MBEDTLS_ECDSA_C && MBEDTLS_ECP_RESTARTABLE */ -#endif /* MBEDTLS_X509_CRT_PARSE_C */ - -/* \} name */ -/* \} addtogroup x509_module */ - -#if defined(MBEDTLS_X509_CRT_WRITE_C) -/** - * \brief Initialize a CRT writing context - * - * \param ctx CRT context to initialize - */ -void mbedtls_x509write_crt_init( mbedtls_x509write_cert *ctx ); - -/** - * \brief Set the verion for a Certificate - * Default: MBEDTLS_X509_CRT_VERSION_3 - * - * \param ctx CRT context to use - * \param version version to set (MBEDTLS_X509_CRT_VERSION_1, MBEDTLS_X509_CRT_VERSION_2 or - * MBEDTLS_X509_CRT_VERSION_3) - */ -void mbedtls_x509write_crt_set_version( mbedtls_x509write_cert *ctx, int version ); - -/** - * \brief Set the serial number for a Certificate. - * - * \param ctx CRT context to use - * \param serial serial number to set - * - * \return 0 if successful - */ -int mbedtls_x509write_crt_set_serial( mbedtls_x509write_cert *ctx, const mbedtls_mpi *serial ); - -/** - * \brief Set the validity period for a Certificate - * Timestamps should be in string format for UTC timezone - * i.e. "YYYYMMDDhhmmss" - * e.g. "20131231235959" for December 31st 2013 - * at 23:59:59 - * - * \param ctx CRT context to use - * \param not_before not_before timestamp - * \param not_after not_after timestamp - * - * \return 0 if timestamp was parsed successfully, or - * a specific error code - */ -int mbedtls_x509write_crt_set_validity( mbedtls_x509write_cert *ctx, const char *not_before, - const char *not_after ); - -/** - * \brief Set the issuer name for a Certificate - * Issuer names should contain a comma-separated list - * of OID types and values: - * e.g. "C=UK,O=ARM,CN=mbed TLS CA" - * - * \param ctx CRT context to use - * \param issuer_name issuer name to set - * - * \return 0 if issuer name was parsed successfully, or - * a specific error code - */ -int mbedtls_x509write_crt_set_issuer_name( mbedtls_x509write_cert *ctx, - const char *issuer_name ); - -/** - * \brief Set the subject name for a Certificate - * Subject names should contain a comma-separated list - * of OID types and values: - * e.g. "C=UK,O=ARM,CN=mbed TLS Server 1" - * - * \param ctx CRT context to use - * \param subject_name subject name to set - * - * \return 0 if subject name was parsed successfully, or - * a specific error code - */ -int mbedtls_x509write_crt_set_subject_name( mbedtls_x509write_cert *ctx, - const char *subject_name ); - -/** - * \brief Set the subject public key for the certificate - * - * \param ctx CRT context to use - * \param key public key to include - */ -void mbedtls_x509write_crt_set_subject_key( mbedtls_x509write_cert *ctx, mbedtls_pk_context *key ); - -/** - * \brief Set the issuer key used for signing the certificate - * - * \param ctx CRT context to use - * \param key private key to sign with - */ -void mbedtls_x509write_crt_set_issuer_key( mbedtls_x509write_cert *ctx, mbedtls_pk_context *key ); - -/** - * \brief Set the MD algorithm to use for the signature - * (e.g. MBEDTLS_MD_SHA1) - * - * \param ctx CRT context to use - * \param md_alg MD algorithm to use - */ -void mbedtls_x509write_crt_set_md_alg( mbedtls_x509write_cert *ctx, mbedtls_md_type_t md_alg ); - -/** - * \brief Generic function to add to or replace an extension in the - * CRT - * - * \param ctx CRT context to use - * \param oid OID of the extension - * \param oid_len length of the OID - * \param critical if the extension is critical (per the RFC's definition) - * \param val value of the extension OCTET STRING - * \param val_len length of the value data - * - * \return 0 if successful, or a MBEDTLS_ERR_X509_ALLOC_FAILED - */ -int mbedtls_x509write_crt_set_extension( mbedtls_x509write_cert *ctx, - const char *oid, size_t oid_len, - int critical, - const unsigned char *val, size_t val_len ); - -/** - * \brief Set the basicConstraints extension for a CRT - * - * \param ctx CRT context to use - * \param is_ca is this a CA certificate - * \param max_pathlen maximum length of certificate chains below this - * certificate (only for CA certificates, -1 is - * inlimited) - * - * \return 0 if successful, or a MBEDTLS_ERR_X509_ALLOC_FAILED - */ -int mbedtls_x509write_crt_set_basic_constraints( mbedtls_x509write_cert *ctx, - int is_ca, int max_pathlen ); - -#if defined(MBEDTLS_SHA1_C) -/** - * \brief Set the subjectKeyIdentifier extension for a CRT - * Requires that mbedtls_x509write_crt_set_subject_key() has been - * called before - * - * \param ctx CRT context to use - * - * \return 0 if successful, or a MBEDTLS_ERR_X509_ALLOC_FAILED - */ -int mbedtls_x509write_crt_set_subject_key_identifier( mbedtls_x509write_cert *ctx ); - -/** - * \brief Set the authorityKeyIdentifier extension for a CRT - * Requires that mbedtls_x509write_crt_set_issuer_key() has been - * called before - * - * \param ctx CRT context to use - * - * \return 0 if successful, or a MBEDTLS_ERR_X509_ALLOC_FAILED - */ -int mbedtls_x509write_crt_set_authority_key_identifier( mbedtls_x509write_cert *ctx ); -#endif /* MBEDTLS_SHA1_C */ - -/** - * \brief Set the Key Usage Extension flags - * (e.g. MBEDTLS_X509_KU_DIGITAL_SIGNATURE | MBEDTLS_X509_KU_KEY_CERT_SIGN) - * - * \param ctx CRT context to use - * \param key_usage key usage flags to set - * - * \return 0 if successful, or MBEDTLS_ERR_X509_ALLOC_FAILED - */ -int mbedtls_x509write_crt_set_key_usage( mbedtls_x509write_cert *ctx, - unsigned int key_usage ); - -/** - * \brief Set the Netscape Cert Type flags - * (e.g. MBEDTLS_X509_NS_CERT_TYPE_SSL_CLIENT | MBEDTLS_X509_NS_CERT_TYPE_EMAIL) - * - * \param ctx CRT context to use - * \param ns_cert_type Netscape Cert Type flags to set - * - * \return 0 if successful, or MBEDTLS_ERR_X509_ALLOC_FAILED - */ -int mbedtls_x509write_crt_set_ns_cert_type( mbedtls_x509write_cert *ctx, - unsigned char ns_cert_type ); - -/** - * \brief Free the contents of a CRT write context - * - * \param ctx CRT context to free - */ -void mbedtls_x509write_crt_free( mbedtls_x509write_cert *ctx ); - -/** - * \brief Write a built up certificate to a X509 DER structure - * Note: data is written at the end of the buffer! Use the - * return value to determine where you should start - * using the buffer - * - * \param ctx certificate to write away - * \param buf buffer to write to - * \param size size of the buffer - * \param f_rng RNG function. This must not be \c NULL. - * \param p_rng RNG parameter - * - * \return length of data written if successful, or a specific - * error code - * - * \note \p f_rng is used for the signature operation. - */ -int mbedtls_x509write_crt_der( mbedtls_x509write_cert *ctx, unsigned char *buf, size_t size, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -#if defined(MBEDTLS_PEM_WRITE_C) -/** - * \brief Write a built up certificate to a X509 PEM string - * - * \param ctx certificate to write away - * \param buf buffer to write to - * \param size size of the buffer - * \param f_rng RNG function. This must not be \c NULL. - * \param p_rng RNG parameter - * - * \return 0 if successful, or a specific error code - * - * \note \p f_rng is used for the signature operation. - */ -int mbedtls_x509write_crt_pem( mbedtls_x509write_cert *ctx, unsigned char *buf, size_t size, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); -#endif /* MBEDTLS_PEM_WRITE_C */ -#endif /* MBEDTLS_X509_CRT_WRITE_C */ - -#ifdef __cplusplus -} -#endif - -#endif /* mbedtls_x509_crt.h */ diff --git a/windows/mbedtls/x509_csr.h b/windows/mbedtls/x509_csr.h deleted file mode 100644 index 674f9ce7..00000000 --- a/windows/mbedtls/x509_csr.h +++ /dev/null @@ -1,300 +0,0 @@ -/** - * \file x509_csr.h - * - * \brief X.509 certificate signing request parsing and writing - */ -/* - * Copyright The Mbed TLS Contributors - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 (the "License"); you may - * not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ -#ifndef MBEDTLS_X509_CSR_H -#define MBEDTLS_X509_CSR_H -#include "mbedtls/private_access.h" - -#include "mbedtls/build_info.h" - -#include "mbedtls/x509.h" - -#ifdef __cplusplus -extern "C" { -#endif - -/** - * \addtogroup x509_module - * \{ */ - -/** - * \name Structures and functions for X.509 Certificate Signing Requests (CSR) - * \{ - */ - -/** - * Certificate Signing Request (CSR) structure. - */ -typedef struct mbedtls_x509_csr -{ - mbedtls_x509_buf MBEDTLS_PRIVATE(raw); /**< The raw CSR data (DER). */ - mbedtls_x509_buf MBEDTLS_PRIVATE(cri); /**< The raw CertificateRequestInfo body (DER). */ - - int MBEDTLS_PRIVATE(version); /**< CSR version (1=v1). */ - - mbedtls_x509_buf MBEDTLS_PRIVATE(subject_raw); /**< The raw subject data (DER). */ - mbedtls_x509_name MBEDTLS_PRIVATE(subject); /**< The parsed subject data (named information object). */ - - mbedtls_pk_context MBEDTLS_PRIVATE(pk); /**< Container for the public key context. */ - - mbedtls_x509_buf MBEDTLS_PRIVATE(sig_oid); - mbedtls_x509_buf MBEDTLS_PRIVATE(sig); - mbedtls_md_type_t MBEDTLS_PRIVATE(sig_md); /**< Internal representation of the MD algorithm of the signature algorithm, e.g. MBEDTLS_MD_SHA256 */ - mbedtls_pk_type_t MBEDTLS_PRIVATE(sig_pk); /**< Internal representation of the Public Key algorithm of the signature algorithm, e.g. MBEDTLS_PK_RSA */ - void *MBEDTLS_PRIVATE(sig_opts); /**< Signature options to be passed to mbedtls_pk_verify_ext(), e.g. for RSASSA-PSS */ -} -mbedtls_x509_csr; - -/** - * Container for writing a CSR - */ -typedef struct mbedtls_x509write_csr -{ - mbedtls_pk_context *MBEDTLS_PRIVATE(key); - mbedtls_asn1_named_data *MBEDTLS_PRIVATE(subject); - mbedtls_md_type_t MBEDTLS_PRIVATE(md_alg); - mbedtls_asn1_named_data *MBEDTLS_PRIVATE(extensions); -} -mbedtls_x509write_csr; - -#if defined(MBEDTLS_X509_CSR_PARSE_C) -/** - * \brief Load a Certificate Signing Request (CSR) in DER format - * - * \note CSR attributes (if any) are currently silently ignored. - * - * \param csr CSR context to fill - * \param buf buffer holding the CRL data - * \param buflen size of the buffer - * - * \return 0 if successful, or a specific X509 error code - */ -int mbedtls_x509_csr_parse_der( mbedtls_x509_csr *csr, - const unsigned char *buf, size_t buflen ); - -/** - * \brief Load a Certificate Signing Request (CSR), DER or PEM format - * - * \note See notes for \c mbedtls_x509_csr_parse_der() - * - * \param csr CSR context to fill - * \param buf buffer holding the CRL data - * \param buflen size of the buffer - * (including the terminating null byte for PEM data) - * - * \return 0 if successful, or a specific X509 or PEM error code - */ -int mbedtls_x509_csr_parse( mbedtls_x509_csr *csr, const unsigned char *buf, size_t buflen ); - -#if defined(MBEDTLS_FS_IO) -/** - * \brief Load a Certificate Signing Request (CSR) - * - * \note See notes for \c mbedtls_x509_csr_parse() - * - * \param csr CSR context to fill - * \param path filename to read the CSR from - * - * \return 0 if successful, or a specific X509 or PEM error code - */ -int mbedtls_x509_csr_parse_file( mbedtls_x509_csr *csr, const char *path ); -#endif /* MBEDTLS_FS_IO */ - -#if !defined(MBEDTLS_X509_REMOVE_INFO) -/** - * \brief Returns an informational string about the - * CSR. - * - * \param buf Buffer to write to - * \param size Maximum size of buffer - * \param prefix A line prefix - * \param csr The X509 CSR to represent - * - * \return The length of the string written (not including the - * terminated nul byte), or a negative error code. - */ -int mbedtls_x509_csr_info( char *buf, size_t size, const char *prefix, - const mbedtls_x509_csr *csr ); -#endif /* !MBEDTLS_X509_REMOVE_INFO */ - -/** - * \brief Initialize a CSR - * - * \param csr CSR to initialize - */ -void mbedtls_x509_csr_init( mbedtls_x509_csr *csr ); - -/** - * \brief Unallocate all CSR data - * - * \param csr CSR to free - */ -void mbedtls_x509_csr_free( mbedtls_x509_csr *csr ); -#endif /* MBEDTLS_X509_CSR_PARSE_C */ - -/* \} name */ -/* \} addtogroup x509_module */ - -#if defined(MBEDTLS_X509_CSR_WRITE_C) -/** - * \brief Initialize a CSR context - * - * \param ctx CSR context to initialize - */ -void mbedtls_x509write_csr_init( mbedtls_x509write_csr *ctx ); - -/** - * \brief Set the subject name for a CSR - * Subject names should contain a comma-separated list - * of OID types and values: - * e.g. "C=UK,O=ARM,CN=mbed TLS Server 1" - * - * \param ctx CSR context to use - * \param subject_name subject name to set - * - * \return 0 if subject name was parsed successfully, or - * a specific error code - */ -int mbedtls_x509write_csr_set_subject_name( mbedtls_x509write_csr *ctx, - const char *subject_name ); - -/** - * \brief Set the key for a CSR (public key will be included, - * private key used to sign the CSR when writing it) - * - * \param ctx CSR context to use - * \param key Asymetric key to include - */ -void mbedtls_x509write_csr_set_key( mbedtls_x509write_csr *ctx, mbedtls_pk_context *key ); - -/** - * \brief Set the MD algorithm to use for the signature - * (e.g. MBEDTLS_MD_SHA1) - * - * \param ctx CSR context to use - * \param md_alg MD algorithm to use - */ -void mbedtls_x509write_csr_set_md_alg( mbedtls_x509write_csr *ctx, mbedtls_md_type_t md_alg ); - -/** - * \brief Set the Key Usage Extension flags - * (e.g. MBEDTLS_X509_KU_DIGITAL_SIGNATURE | MBEDTLS_X509_KU_KEY_CERT_SIGN) - * - * \param ctx CSR context to use - * \param key_usage key usage flags to set - * - * \return 0 if successful, or MBEDTLS_ERR_X509_ALLOC_FAILED - * - * \note The decipherOnly flag from the Key Usage - * extension is represented by bit 8 (i.e. - * 0x8000), which cannot typically be represented - * in an unsigned char. Therefore, the flag - * decipherOnly (i.e. - * #MBEDTLS_X509_KU_DECIPHER_ONLY) cannot be set using this - * function. - */ -int mbedtls_x509write_csr_set_key_usage( mbedtls_x509write_csr *ctx, unsigned char key_usage ); - -/** - * \brief Set the Netscape Cert Type flags - * (e.g. MBEDTLS_X509_NS_CERT_TYPE_SSL_CLIENT | MBEDTLS_X509_NS_CERT_TYPE_EMAIL) - * - * \param ctx CSR context to use - * \param ns_cert_type Netscape Cert Type flags to set - * - * \return 0 if successful, or MBEDTLS_ERR_X509_ALLOC_FAILED - */ -int mbedtls_x509write_csr_set_ns_cert_type( mbedtls_x509write_csr *ctx, - unsigned char ns_cert_type ); - -/** - * \brief Generic function to add to or replace an extension in the - * CSR - * - * \param ctx CSR context to use - * \param oid OID of the extension - * \param oid_len length of the OID - * \param critical Set to 1 to mark the extension as critical, 0 otherwise. - * \param val value of the extension OCTET STRING - * \param val_len length of the value data - * - * \return 0 if successful, or a MBEDTLS_ERR_X509_ALLOC_FAILED - */ -int mbedtls_x509write_csr_set_extension( mbedtls_x509write_csr *ctx, - const char *oid, size_t oid_len, - int critical, - const unsigned char *val, size_t val_len ); - -/** - * \brief Free the contents of a CSR context - * - * \param ctx CSR context to free - */ -void mbedtls_x509write_csr_free( mbedtls_x509write_csr *ctx ); - -/** - * \brief Write a CSR (Certificate Signing Request) to a - * DER structure - * Note: data is written at the end of the buffer! Use the - * return value to determine where you should start - * using the buffer - * - * \param ctx CSR to write away - * \param buf buffer to write to - * \param size size of the buffer - * \param f_rng RNG function. This must not be \c NULL. - * \param p_rng RNG parameter - * - * \return length of data written if successful, or a specific - * error code - * - * \note \p f_rng is used for the signature operation. - */ -int mbedtls_x509write_csr_der( mbedtls_x509write_csr *ctx, unsigned char *buf, size_t size, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); - -#if defined(MBEDTLS_PEM_WRITE_C) -/** - * \brief Write a CSR (Certificate Signing Request) to a - * PEM string - * - * \param ctx CSR to write away - * \param buf buffer to write to - * \param size size of the buffer - * \param f_rng RNG function. This must not be \c NULL. - * \param p_rng RNG parameter - * - * \return 0 if successful, or a specific error code - * - * \note \p f_rng is used for the signature operation. - */ -int mbedtls_x509write_csr_pem( mbedtls_x509write_csr *ctx, unsigned char *buf, size_t size, - int (*f_rng)(void *, unsigned char *, size_t), - void *p_rng ); -#endif /* MBEDTLS_PEM_WRITE_C */ -#endif /* MBEDTLS_X509_CSR_WRITE_C */ - -#ifdef __cplusplus -} -#endif - -#endif /* mbedtls_x509_csr.h */ diff --git a/windows/sodium.h b/windows/sodium.h deleted file mode 100644 index d0bb25c8..00000000 --- a/windows/sodium.h +++ /dev/null @@ -1,68 +0,0 @@ - -#ifndef sodium_H -#define sodium_H - -#include "sodium/version.h" - -#include "sodium/core.h" -#include "sodium/crypto_aead_aes256gcm.h" -#include "sodium/crypto_aead_chacha20poly1305.h" -#include "sodium/crypto_aead_xchacha20poly1305.h" -#include "sodium/crypto_auth.h" -#include "sodium/crypto_auth_hmacsha256.h" -#include "sodium/crypto_auth_hmacsha512.h" -#include "sodium/crypto_auth_hmacsha512256.h" -#include "sodium/crypto_box.h" -#include "sodium/crypto_box_curve25519xsalsa20poly1305.h" -#include "sodium/crypto_core_hsalsa20.h" -#include "sodium/crypto_core_hchacha20.h" -#include "sodium/crypto_core_salsa20.h" -#include "sodium/crypto_core_salsa2012.h" -#include "sodium/crypto_core_salsa208.h" -#include "sodium/crypto_generichash.h" -#include "sodium/crypto_generichash_blake2b.h" -#include "sodium/crypto_hash.h" -#include "sodium/crypto_hash_sha256.h" -#include "sodium/crypto_hash_sha512.h" -#include "sodium/crypto_kdf.h" -#include "sodium/crypto_kdf_blake2b.h" -#include "sodium/crypto_kx.h" -#include "sodium/crypto_onetimeauth.h" -#include "sodium/crypto_onetimeauth_poly1305.h" -#include "sodium/crypto_pwhash.h" -#include "sodium/crypto_pwhash_argon2i.h" -#include "sodium/crypto_pwhash_scryptsalsa208sha256.h" -#include "sodium/crypto_scalarmult.h" -#include "sodium/crypto_scalarmult_curve25519.h" -#include "sodium/crypto_secretbox.h" -#include "sodium/crypto_secretbox_xsalsa20poly1305.h" -#include "sodium/crypto_shorthash.h" -#include "sodium/crypto_shorthash_siphash24.h" -#include "sodium/crypto_sign.h" -#include "sodium/crypto_sign_ed25519.h" -#include "sodium/crypto_stream.h" -#include "sodium/crypto_stream_chacha20.h" -#include "sodium/crypto_stream_salsa20.h" -#include "sodium/crypto_stream_xsalsa20.h" -#include "sodium/crypto_verify_16.h" -#include "sodium/crypto_verify_32.h" -#include "sodium/crypto_verify_64.h" -#include "sodium/randombytes.h" -#ifdef __native_client__ -# include "sodium/randombytes_nativeclient.h" -#endif -#include "sodium/randombytes_salsa20_random.h" -#include "sodium/randombytes_sysrandom.h" -#include "sodium/runtime.h" -#include "sodium/utils.h" - -#ifndef SODIUM_LIBRARY_MINIMAL -# include "sodium/crypto_box_curve25519xchacha20poly1305.h" -# include "sodium/crypto_secretbox_xchacha20poly1305.h" -# include "sodium/crypto_stream_aes128ctr.h" -# include "sodium/crypto_stream_salsa2012.h" -# include "sodium/crypto_stream_salsa208.h" -# include "sodium/crypto_stream_xchacha20.h" -#endif - -#endif diff --git a/windows/sodium.lib b/windows/sodium.lib deleted file mode 100644 index 8b374021ee03455381de63fd58f2b16be855b068..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1920144 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%0D0VsJx{Nr&G5w!Ayjz z&dkfsD$9VA9`6G;`s6E^B24RuPM{ zquD*TlwQbPpQ`ug?rfPZ9~;VUzh&H_{e9z$;+kiTSv7YR=M(c9F>yK~Q0;o0oD=8I z9P>7(BEbO8UmL;mat`BvoR!scYsretF{`qAt}8t?d!a1D=Ct2<_QHlubK0+oZ)v5+ zKOamyvZ}K`eL*lGbK&_u!(6h*-cXek_8#iO3gqoyadEykJ*=*y!(v#NHb=cZqhJ~Dko?jhs_@E81+jZO}_|E2=b-tC4+w&RtD+T`~cBsEy6rYQAAEUogK3taN^I5!t z#rG+iB?OCuMKM0lGqL8GUiTT^Ig(~D^5v73`0mGX7y27TZK=~Wir<(Uta){678fjNY zCIEXt@_Dg&7hjCEJLX^DO;2o;gaW zLGJsa?+Ne$V%|rFr*F(xCX10cCyGE@JP4egcp2hJY)E`4D2R^@vA_~w<-&7;@RTE- zEIe2EKEb`A2y{Irx%}ojZY7Y+%fRyhA#1Y4DbUBf;R8?3QJCq7o3DJY$LH|U6K5m7Nm_qt zL>+9vud(h7kgrUIT%kf*VM4Zw@q2|d-zlY#Ric5qD&$lVvLS?w6QQWJ*F>l+5sE_Z zk~E)*P*V_plrmeN1jhTeTLb)6!go3FQ$^cD?8-VO6D?&;QK85C)}br+?~uPNh$eU{MdUP_40FBya6xHy3$XKAJP~sl8^Hb#EpDLRhG3XntbPplvL?v zNp~eaOKm#nN}Tn!?lDb$+o~#V^ZxFQUKL+ zud2bRMo~!tzQIj?<|8-x;0bFd{cgA{enEznaND+Ch?>%n?iu{>Ww7Qap;tW?^L~h$ z%y%)_4LFYBc@lV;om434EyVtrbZw*9IHq zr`fl*)m4F%UUjAVi~N0JW32gc_e8{N!?DKpr(G%3WKqe-QV`pqpH$hh&D`m8r#D|= zO^wFDhf=F6Cr?jf;=o#+Ewv=5vl5}OYXQ-ygF&y^ZXLwptU#Y;8VQD-1NSB&HS;1DSH z0Bn*fou(=*mG}(AjT?O(ac?cmS9_--pDRT(tMG7Qdg4?~AK!eHRT8y4dScAY_0JA; zZM-btXb2wCT8$q*IoG@qDtTkfr{bDT|6wzjul7>ra>@Hm$=j8!Ve<;>*{GS(566tH zKO+#`_;WXJ=;cGn`^x5rp#AcrweMPWK(uc@FiabZl)`%ZTB9PaEu9D#+RQ5OzY%d& z^xZM%)?XC(vC*$uAa@milmaI--;DxCqCihqJYBouZPH@(gH5d!zHfXdN>^k{@>60Q z*QtHF*GDCOE}XKX?dh;ewiykkr*xGkf+t(8e3$sL#JSS{^_Qx2kX+a%dnJBnSeqHc zW2s+=*EwEGT_#mGez9EJklz|N9)~vY2GYTMHLr^{n-=>6hJQywwpkD5<@Chc-TH*8 zq^nk@tyJdPKm$Bpa-a33dZ%Tgf=U(cB8d<3>b1LOa-!(9X=GDtw*N){MXHgWQlk?j z6S}zj&i@m7kfmR(Oum;zp8_- z=?_}>NL|=4mq;g@DjBd-@=HT}hSh2A15&rMB@g!ecjbI0Ue4kZA?SK$Nulh3>4|d? z=Qn%ste%v?;oo%7^R)qX;{352V20~{z7;DlAAY4(d@32I!>jZVuR?o#j%8WHeRTD- ztX;zMxvvMfC=i7{UMkVcPUHAS*Lo^YDX&27Vt)&>9^9BQfNTl5ra zgH(D`pe4SBINUc$Teq$VYsN(?=N5?%#xtN!*O=bPbpf}Fyc}=*A_jANu(Q>{8f|s(KkPrt*C%M^4m3D^_Q$fKv^{%^ zh{oS=6v&tOh0?wpP+umyH7z*QcR&QVOa(Ym;xkpRw`d#fL^^N##>X`D4qH8>vr^dp zBJ8@rN!HWW2iDiBfp(&S7Q#oNi-hl`h)b=Ffsqh}hQ>~Og0#av5ro4-oEY#BiPM8$ zA^FUgc&Q40h6-AZs(A%H9DPzus40ZJQ@1baZJly1w=oubGOixj-RUs8Q2NSbLeV_Wqt0Fc^Yg{4~V1*+j z9^(4CX(n(qI9>^yjJvd>eNSs9BPEfsN@{bZ#Pd9^Z$>Bcv1YbOo`^CQnGKX2*y-~` zkAuW3(XQW?_IM3XCVwwHG@O46&vhayoj|_iWAdPv`jo<1ouutJEihi{3gMrlMzl)X zh8IKLw&Ac5HgPf^_GX71CjzyPyv({EU%KGfh|4a)A~eGa@ZE|ZKIhmx9gQ(QrfZD1 zW4hDir$qEI5zVcs{zv`Gq>9u&Kk&n;CrhAOmP(a#ydBYe^_+U_CarNRo`wt#%2+KhQ3MMfqNBLPtut4G-LCSW}f;wer6YUj0bM;~)pI>FZ1-+Nhd$ zB86AzUNL*ZX0PUYU)-HEvwPjn$K>&%4H?{2!e%)lte2?AwVj32&M8dYXp5y@#Q zbC@dD)F9Z=itu;yZx@T;>}ajzG(~cz&wWxF%$%nW;aA!)cu9TO#f{?4G?zZaG|mOu zIMrbm<&SL@);Q+koC95RX^eJ0`geCwYb*Z!fYmGN!kCTGL+W1>__fiG+8{~De=E{L zvgQ$RDWAv8@m_el^Yj8yWn=jFpD|_Frw!s0i!eg4LFP#OJfCaw)7?oazjKI@1f{RT zSD?*&kNS`CWWs*QFTe3S$6?;m?K|g*n$3VfwJwqPJ-+6sy6cL2OVv;I)s5N~JxH%- zt09D6n(yM~@4^OiV1sVMPw(W8X?fCkmj@eKyL?Ujmq?u{Zn~6~B#lgQ`Jpw-y2F#p z49_L;Jv?dcR_AoKDB%$dPt}qm&E_)6AwziB@6Q#U4MhJwtMdnI$m6Z&IgWelCA`$}|@|WGED&PV+Uf4tjIwp;9Lb zLze&@t0#wU%FsY$ve1<_g-GpIoK76mYG|g?n}wNxQHbw^A7uJFti!8xv zrjy4q;aG~EQYsbUv)a16eJ6rDWxd`&~hV-7lAIBLY7YfD2=wEV8eqER+`K{Dd>-)uD)<+=IRgmSsQF& zJvuk@CuV%VxSiYJx~9u?l^K5`8gN~gdADgV(Z&&+?<0OQM&c~s1>1GGZf3f@&chP; zrBu9O79~Vz36F|?%r*d>;|&rwbH-}KkCzfmtFpu>Y1Igi*AaoS)g#v6oe}dx z(U}?jXjR*!MdQ_i?ve8e;!#+mPNJ2`ktbNY1NDM2D$wJCufcrFsr}JoQd*cJ4ej}6 zS(-!2I5CFJ*uLcu<5cX;gcVkk=%1tS2rP~}tX4P*dA1dqyCk*=wL(|Z_lQ=PqwCxx z&Y9iiG$~{rx{I09QTt1UFCCV7LXZlgrbpNv)oUbf*IytuT83HRS@>~;!wmp}b&H6$ zE)JFkhpH+Y;>HoavYk285BWuc5aTcxy0qnHcDvA6m}x7~Z@q|))*C?}{mqA+|$%NUBgx=2HIjHfMLl<;Jr%4@}D z#!o>@MfrWA##*c;eWW}$R=1UZfveMZS?rcsrasWt|~PXb-PkPrb6aZL6o46O~Xb)w&pS`c@(iq_eJ< zbdeHA=GMP#PfZ_lm)idy2@R~3(Wl1b1>TC=i0V2slNnvbp4jZMqKko%{jHaMvFLis z{Ee*65tg(XJL+<271JVY_BW+UH>#t2MZ5zyr;KT^aHG4IBoEi&J*KKjLq1MtpJvJL)9s59x69<{PEf7s>TaTVEp)g0|X3D$y>YyR}z_7z&;yg1d9N2PMB1 zD){~|!TDUZ5&Uuy+@1HilUOQwfwXgCkaw84ZR)JA)A(-|qiri@NU4996eA;sSk-~U zxFwn9%Xa-)_%cPJYQ)=#M6c_$F=M{j*yxTWPPIlv$H&YGJR7$s;Fj(zZL{?6q%m%YF15av8zO0D zcSxgMf;9qj#l?nwLnJs`4B!yrGevQ4Ms2em30`Gisokk1o4r`h+>QAi)GM`}e>z4O z*F%5yrsy2z0K5$g&ejE3A^kxy`JW@R#a0cH$Ay-{5!Gpw!VMiUQy$%ccOr7 zQA+}b@KA4UmbhsEUkb-p!2;_C-=%(Y9{@!^Q5w9lih8}2Iu3({sSb@`h=^)t>u781 zU5{DP7%mjE3~EoV2y1+=NtzD9lJ|;9J{LXB`atf-Wf!|oq&Zd0j~!sKNPTzIvsRtp z274G}N7ybUUa7;NJ8qb34qh=ClUxu!?D3`sqqsC=G(rln689s4>lo~%Z%XuuFf{a@ zG+u-4yiQJbL646|+nQT+!xpFK3QMNPDL9MMY?X6Gv$tY}l`8;_+prg>$*@!^S{-O) zm3juj64$LfE~fU1uYq-&tgxEpkP`R0FJ6jx4-I^Oh!F0~`5xhT9<%058G_iPMtO0b zYTJv`6l$Uf)h^J`3V4NbJ7Ka2H5N1cGrdB&?slQ%5F2b{?eym0&cbRWhe4=T8(HhJ0y$Spqa_uEZmN|MY1>x9kq#92)Ew(!Y~uFh9O?%yK~J9VVLS~fVrk?!_ML~ zHPKO2w;dH-79yrA1r`c$w1b#_k85@or zIi9x3b7h_{%)L=;6S=@*g4+bEi(T^PJ&8t}br_0pqcAkq5+0Ni5!hQ_a_e=Coa~9Y*{`Aqc7qBjmD59~Z{pRqxVzXH4Ax2k z?J)28M0^%?Q6yG3I=5wiPoGhBL>f1&1WT|Fn>piP3 zLaY{>DOU17-PhJ~F2(UYgQRaLYWf2_Ws<(oKNdHPPx9q@3~#jSf&UODTI;VKhCJ6B zJ|YY$(j}eizvrZM)2G=WFA0yCJC-9Zqlnatd{1$KSbI?4V7%shTq)h-mwsE?}s zT;g?o5!PW(CZpX_sN zxvGJ^WcWlI?4J(`&nq&ZmLtyQGLzZ~m{hEkT2zFbHS2r@I2$E0r-Ym}u9%JXm83mW zoBm^o*OlpoQ49)CE$*fxlx9LsH>r~Qv9TGD=iXOOa*| zY`f5`O)Id|WXkDLp{HHlES1wIgjB~|WU}PjgX2-w3@1M(fIJ-=9X2 zuCvRe&o1z2;RXL5t7l|UNJllArKFEYs$@?$@r?e%nt}szIG!V~8RID)X3yx~Mwm~B zc=76rdk7YR-tdekk!SSp5>~>QQbO41&*ItR>dVRc+N(a3cm{fvF(kUyQ<_sm(~Ivz zsGacCG8-VXPfoUkv#IfjXU_)c)EC%w>R|0q=l!B!oY6#hhL=2J8x*sSM+Ca7NLvJR z<&693B!giX56-a8w_dOs_^m;FM_K+>-OOW;qF*K^9f#lcGKg(K#TanC6fnMnjk>h}Pud8tx$ziM- z_#)xIUT|X6cT3@&q1a(ye~66@!id=slYFl{=B_Q)b+zR6BrXK@!ymq-ZCoSyy@mWr z1*1IMg*Qpca8|+37i_u=x5c)MhroNh;9dNuTMvs>LTy&GU}G`)?jjGAafVZT9GI|cYJ?X($IqUi+X6ZdD66_v9frY&PFa1zCNO!uGU-VqGsKl&HisW z*A(|^{U+sdFdQd&oF=Bh*+^62>F2-58jky6a-}S;q2THfM=qMJpXWNpGefyW7><)l zH9{py<@&uiucWLXZ_;SINt_>a?wgo~J8k&vt%bYso%_s)%@#RkdOMn1`Lm5s;&B`o zHBg$wOm0|_5yInLPsua$1ZwIuIXM7^WI6^r+2w-8LO<>$Mu}t^<83xGP%x7zlE9Qm zbzUyay*#=rquL z{JP-IuqsegOFoT@jh>1}5l1N%Pk}Z9y&5Mwfb!nT?x3wfhk~90It-LIoL>yO2y{3o z@828&S`9i1l#d|ff$jsH0Ln*O3qb=Yy9hK2v;>s5LQes04muT-kBrO!<#S@QLHUr@ z98f+)btNcodcF#j_nBS|%6mz#0p$~6*Ma7NUJuI08*T*UoxQh$@+suoKraWq6SM*p zeV~GmP{S50_*nG)pnUG{Ay7WC`7r2XppSv_nS&=mSAjkQ%4d9^2jw$ZFM#qntCv7| z6Ywjbyx)B_DDP>19W)7a9Vnj?78MHBIDrj?1K9_U?=y1@MpyNSXgH8lJ8FU8dDWF$@b_Bfzv@_@u z(5|4%K+gbO3wkE#JD`0)w}bWt{TTEt(Cr5EQmdJF*r-U;ScDpBV%i4;@|tivC?u2zmnORiLdvc|&Ap&{61{y+Fr+o)5~s z>cOCMLD^TX1I-2HJ&Noz9YEP0wJwX?P^1jS_K+gqbUppU^eQh}CgP@~8 z9|FA$^kL8`ppSvh0eu{lHz7U=dJE`Npm%~k1Ns2yGEm;N@;vDCpzMn;fxZN~4s-=5 zw;^8v{Q&e;&>f)cn_q&y2FjaUUk5!1x(@UZ=o_Gs^N?&pa^fXZ1ZX`Y>9<&SM z4M6EzT7z~2JsFff<}}b=pj|-eYcfIUYjQy8YlebmfL;trpECiJKBowjK4%(eHs~x+ z`ksZL^gTC%()Zj2Is~*Dls@QbQ2L`>9w>dx4$#e@kDxuFKaNe-_rO3Z z4uG;vc{$5A<+Tsnl$VoiQ_iE&#Vgu_!elCVb-}jfA_&`-b97X{A|3R2&^%DKZ9Zrc zXaQ(D(0QPpLH`AMIw;o*x`J}yp&RIGP}p0=>!9hNUxKntzXrVk^mNSKF9PiW$~Nr> z$~GMUIui68&|FYnD&>Pt0A-sNf?f*BtDT9UQ$X3K6`*X>`JhFhm7te_E(Dzcx(sv{ z=yFiDXy2NYYQ7%a@@+ost2ihZM4jbiZgc76@)P%M43jkQ;7 zuww9hb{P{DD^iS)d7JvqQS2tgZc}WzVk;C|uNZrS(bdO_?Nsbr#r7%24r9s<+M!r^ zPRw8p6-!pWwuH?j(pbHV67E9S^0R4W$MPOEW^j^D}(V0%U~BPc8OvW z6yw8LhVM$nu2Sp~#rS4c!?#?q6^dQ$$75hT5eTx00SP)Lp$Q*5lVy$+H@qsl{ zZn|Rq6+2I{p^9CmSh-?XC^lcQ8x*@mu}2kqLb10L+ol-bd}UtPUK# z;fuFJvDR^lwNy;5*U+2mSltvGt9+L#R;JiA#pWrtK(T7Y9#m|DVw)A)q1b1N`6C^f zBkfSE)mX9Sik+<3DT;Mf>Uc>Vugy8D0ZD5vC)c+Q*4T2GZedDv4<2}r`SftK2>a& zVl|5K{XVA8#Nm3~)FZ(T#ahXVwNsVlxz*t=Mgf-KE&S6njds zmlb_o-dD3+;Mwqh45HbSvUiWMt1 zQ?Uxg7AbbKV%3U0sMs>a`27&mZXYYQQ?Y%D{iIk^td*GdZDEIE<-rSsbx@2q1sZIS zVwWjau2`jF3l+OVvAY#}L9v$=8GKxYp;+rv#daxHqu74M;^LgN33e#f;<}H? zFGaClilr%*tr+iAGv$s@Y?NZfij^r=q1ZgdZdQydM<%})75lehZ!7k$VxK9tTd{+R z9a1a?YbmB~b?s2B)l9MD731xwhL5XAMwZ@+r7On!*9>2QV&#fWSL|-Z?o({FVy`Rq zp<{gk>Ojb*apQuP>d@_hR+}Glo4r%Fm@}}T(MIW6|B+abkbu<&EB zx^^hmI$p6>iuG44OR=GfU8LA7#ja57e#IVA>?OrsQH;-57+LtNhtbs@#lBJOh+??Q z+UAS5LzvSjmZDe}#nKeZP>i=@nR16HR;(EBb2PHts@R>1J*n6;ioK%PYQ;WQY^P%1 zDz;BCF2EUCf_4b&5sEcbjO(U`k1MA}zV3?kQjF`ThHsE!qZAvf*bR!^qS*6_@fJT* z#_Ni`q1axR9a*<2tI5k1MH$Z-`^8-gDYjg( zcNP0Uu>*?zu2>zcn;Kc-?GWaUigi#dOR*fqCMZ^@Se0VeEB1t9yvNSe_f5sP>T9qX z#r7+ffb~-&Uqd?-Yjswvt71bGyHGLS1Y*+8Q0z9v?oy2R>li++?iyKMR_s;9-coFv zVxKCuOR*Zo_A7=*$?Y=g*dgdyu?~uLSFD#}nTlmAcA;V;6uVThNs7%>j4Q^bZZ|7- zyJ8P2_K0F!J2rY*q1XqC?NID1#lBPQpkjv!et3#kc~E z|G2K;+A~9Stf7h(C^lI!u1(`V_^wy%F2(Lu>|csKrP#}gy{gz-ifvPDmtuPqJD}L_ ziZ#XhHzJUwg&o2*j$$c_^-?TNv24W#DK_x@a zDz-tfuN3=Ev3l6ZGBVe>|ZRDmFv0*@|7K*o}(a zuh>J1J+IhHifvQueZ}@D_Kji(6#HGVs3fOub?i{A)kLx56gyq99*SitmZR83ij7pP zK(Wb+O;>D|Vz(-Gr(%yO_M~E~6?+ z)>E-Qik+j_K*h!=HeRu*ie0W)m15T`_LO4JD)y>kYZUubv0aMcJF!4VgVi}4JQS3s+#wvEHVr7a=Q|uPS z?ojMe#hy@XqhfC<_NijK6su8ezhZ%to%%-EA=s{BO%yv(u{MfjD>g{6Ns1LKHdC<* z#jaIskz&<~J*e2ziY-&@HO1B|wp+2U6g#NcA;l8gIQ4C0hcM4ktes+~E7n7?{)%NO zc9CKu6)R9|vSPCoyF#&h6?;Ijrxbfuv2}`VRP24lK2q!(#cCA$U9lsI)oJVKAl?pP zJzKF873-(i0L2C?HcYXpie0YQ?TRf{>~Y1OR&0f0D;3+M*cQb;R&1wYzbSTDv3l*C zy4ANsm=h}2TCt9bbyn<5#ri6Cu43maHchdaie0PNBE_l|dr+~b6b@AHeazy#comT4#k!z_NZd36?=woDQ*4Q1 z&nfnzVrv!KpxC>LeW2J^ihZY8oeoaj;_VRnrD7*4cA8?TiVaZg9L0t!maEuA#flVL zq}a`hEmiC}#oknGt71PW_OoK~r#kg$V2ALmlVV*IJ5RBpij^ofMX{R|yIrwI6nk8; zwTf*}>`TSIRxGNcBVQdmgtaQgIw*FwV&^J0QL!S$u2pQ2VoxZxRI#@d+osqLiv6ru zgVP-On%JROtBYdY70Xa8Q?We7@)avrY`S8%Dt4!04=MJTVrvvzr`Y?7eWchP#lBH2 znBvqU+74l?Rji$2JrwJ$*u{!nqFA9~C5p{f>`KM%QtV#Eo>J^t#okhEn_{~Z+oRa; ziXBlbzLQh826hPdN-EY~v7US z75iSXI$fN4#M>c^0g9cd*lCKTDmGBD!HVT7Hb$|_6`P~j^@`o3*aM0^tk?^Ry{y}ADX zRqO-Bb}06hV&5s2(AANzp&g2~Ix5y#v24W#DK=5DBE_yz>^j9BQS5QWmMgYGu}>A- zrC5z(`xR@{&5^m89l}17VqF#MtJqnJg4RK>a}HbAj+6uVThNs3KVY^Guh6uVZjrHVbL*eb== zD)zQw?<)44Vm~NWw}(@YL_383q1Z`^^-!$0V&^F~RIz-;3KW~J*eu2FR_s2-o>S~a z#WpCmS+O08eWuvYiXBue<_xE9b?p%DNl>hnVrMGWSF!UIyFjtgij7ljj$(5at5)nm z#g-|yT(Nf)+pgGN#lBVS55@2~Z}CzXXY3I6ycFx8Sc+o36iZWVh+-EiR<77|#i|v1 zP_bo-Em!Pq#okqHw_;x@_OoIK6|3LNX`{w=DAqbzu~QT~L$Naz8>ZNB#l|Z(QL%Z7 zEl})s#TF~}h+>Z`wqCJKihZouPQ~^q_LE}4-cH@3?GV~cu@;JTP%K5UUW%nDmaW(z z#YQPMR2(r!KM&tRvgQj8@C_z8H%PNCx8puO?S|uc4=>-u;r)F3V|Lu-<(rb~HDo$| z_&(CgNK8UF*~&>I#^V6$8WuZse_)G_ z=ARQH{egr!Z6(fR_3-WoKQiYVp3g$HT3GcHilw-t{-m4tKa9c&zW;%5cc?&^dgj|5 z5)og3ICY7*(Pd)dB1Fv#A*ip!#M?DKQC=aDggEPMULoPU^MPE>!S?0>>G-U+97FJ& z85?jIC>F)!8NBhJBM>hE9SK@)$EVxz%kB6q(BU{g-;OV|<4Zu9?lI6yK>rPz3(D5a z1Esx<2E_sZjD~yvDRwB9&%E0hB$AMM=FRX?GKOOL+@;NTm0~w4cB^7fD)x+GuPC-! zu@4pdM6s_G`(80VA!cNWwnJ_oTMVRoSReZwDI4MU7Jk3MG|b`cW4Y6cN^%Rv{7F|k zPHdcg=Qv4TVaI)Z3KT~pPyd)K=h*L>%DD}iR-!a_nKba(b;c6aAxFYNy8>!$nLZRG9lg?K1CF|k@%MAwOl z)Q8FA&-}sD;BpSyCJ&nBgAy{&*3!xf(Lb@tWufm&KfMeT6KIL|0cE_O9iIX^6><6n z^3MSUR&kAuvo^%(i^zAE9sd_7`Chbf_RDFA)AVM5vd<7huRt&wuE+fJYVP8S9 z){1pj3~wi}%f)*F?6i18fQ?O1tWdG1;crYmmf0cPKUUt-8vbjJHF)p2);Ze;TK(1! zvU)B*&q{sde5?KKgRPSn46#~G8)_YY=>=A^E-&I8-A`Ke_a*GCwkmw_K`B7h%81~Z^bxHnQx< z@N@HVMk9dYQB6MdttOw7CDg2u--_JoPa)SR{8)mUhhxD2MH40u>cr$hA7Z2?2NWj< zlqH9I#>9_xp7rAqVF= z9aB;k`K$_qbR1ifKc;L-KG-F%fDg-}VMoWY`J>CnCv%82W%#zE2&XoN+2+BpZe$4f z(4)eRj$`x6@@$MU2l%p#uzYyhsW++L>R6UPtt@#myOz{4;A0E2zCT{_?IBmYUCDqW z!b(C-y{V{=fRAm)9KXEn-YeWxeSz?ZJSr7681T{R>L9du`FvE!hmcnrw2FW)?rgkA z!fvD}k<7~zBRm|V8zRIb;wYY2paI_~G~NlqL3!*9`9>@INV0V`){aMs%#6_6n8V?z zascr7t;zY5OY_Sdh)Bn2Zey^10Uw)=^?U444?K?Pl>UI7NO-J{#cnG0D%72S@k>oO zaKD>sm7U6*^_C_5%tuYItar12U1YSu1fTE^4$~LCWz_@fmUZ@-_bzr*{U~YV?GkKv zPrc$L6>W@V<-Xn%9~*b-MeoQ%ctYwWy;$qjqV>RvY$-%r4UJbtSPu`wmni_3~~?Q6Tz zyh){bol|nBO`hDfps-A?^>Pa)7f;HcoL^X$S5{C|Xn8LKJz;0MT3R%=pnP&JEP6(M zNs;B`m|rL*<(ik$<>n3;oS9cRc2a&xzf1GSOtf-yOY_H+mlTvu&mBV`cT7>y#DaV) zxx8>{LE+fs+OtU(b3fKeC#mPM)k)xmj%IVJax}Md#w)H7G?zL{O9?$Xg+^0SP|sS} zQp4CfgKhfJxgEt8maMkYYRRTsShB8R@zcZNT|%XCk~_&9sXE)qoa9~XxJqTkp4O>r z_mm{}=mg$Qr+4Yxt&@u}gd|!^=kC@iA%ZRMPGJHBZAFtIf z%OBx%5BA%RMWZL+wMlraZWO-ZD3ka?Ty@jEkmeMODJd!~8dsLo_WX891B=TFCKt>o zC>)>E50jAs2!WTe8ix!ba#P)r_%_(r$bxk#u^Kg6v2N;vea^Wot@@7BA3c2E1@CYO z|8Lhx90^95(bkVY{O_hpG=|W_YX%%TYnv)j53uKM-C(AA{FOa->)NH8VWvGd6nmc1 zMBalW64YkTe9vacQHwqEeKupj*T)__>Ew@P&x4%-*3Q)Mx${2496dLJ@1v0X(pGxs zY~-7Ta%;dd$)51iP?8b`jSA;G1+uLG&zGD=;h?bypJ(1md=U6rV!hWwbBx_ zUk(-!f0Cj7doVFrO`FDX3Vy=pnOhL&1<5!Jbysq@!@;hcmXMO>ow*Q?!|z(~+-~zx z<-!-1AIDlg_r|B&cqpG|)N$YTO{ zuC)37w7%fE{V(!MLVnMK=e=;gTI%}~d0@*tv>svdvA!pOr(-x@E%{}G=fW^P^qv2U z-+chZ=KpW~Zb|;Qf9!U}*Up2ir(1Os>(JKZsC|Nl0>kHJ@m91fvXx`v*0oD+nbjTu z^Qk?DuvuMQJ9TNnjC`d131!oqXhI3TD=uqr+ZAV=)2hMg*Bw~$#ojwC86(D1zAY+HX=kcXA~QO$~oa&YIaQMIJ6htekF>gH>tyyZ%W84 zx7Ic`(Tn{r$0lTd)WJ(6pG;I}^gb)Iazt{|%*wLlq@kG$CnZ0cdBwNYIFVT;E;cju zwam)?12U^V%&ht$bN<%&c}Y-wR+aT4U21xKf)A(PuUdsrRziGY|2^@EIR`T5{+bX! zw>cPJ&xm;6IodKedUVenk-4y*_I*@B#%PWbZ$ zIiESTJ>FI_QLXYIhB`h&pPl)*2J|dvlq^KA-nx=*@2xjfpO#FThiTF zwlN%l-XDglUl65bF6{qHX3t-z*3Dixun!_HWcOTOT0e7Pzdl5J5zX$ou_TqJ(wC

jMb&ktz^sp29#<8PX;?}COgS#$UL%6EMOGk**DBxJVV_!4I-;}-3YFG}0D zKRv!EJ8B=8lg7o@tlu9$C)>A$XLg!1I~9eMe(fuN1D5%8=G=9(*QYY)u1PzxDr;^{ z+Jc;?96MaPAm?c({LF%!rB3+S1v$?+;bjYQo^`^{Ey!6W;RQL*fm%7wXU+Z27e9Z; z;Uh<~VY}53HoIzBGWC;P^?345&`iX)gXYZtB7WZc2pGE#)n)3|{yfc|m_GN9q>_^K zg&E05s@7*O40fmfG6Soq)EJ}Ym8IurE(|s!5|Q0=O=(7Y)ocB$UYonKWk&mz@hvmp zeB+DanlJ2|0GiS01*4aujEEPYv$!+nu1@0lrnb7`x~KQtQnER{=R2i6t5&mlGXv|T z(DvI6`8A{c=g5DY+kWxS2b*>FEeQ6=T$tsPR{PHBhsHRZet1AkMfzbArypis@ftfs z`iR`&BiFEprQsK}cH*z;ZynI+CUgiP)642$%3wmx5NakVg80kJraQou61IEBndeD3 zN4mwH1zLDHwz;mym?1kyNtkFJy+Pd`>oe(zT;XYt9QZcg0r*~myZa>@t7PG3SzBQl z>O0G4zT)U^RJWBP%;r>~Wa;jf&Gaq8OPKk|6z-TUl05cc$+z5EU?t&op!GayCrVm( z_smVp9W*Cxz?yE&!)7(payRW&4lDR}66VroADcU6K-nJZFk< z6UP)4PZ#-8khVFt!8z95fXiS0xE0RdmuNk1iC0VhNByz4#24WNx5V)wd!*%&keq#P zSAGr&-$+1|P3`QFv!9)apJ$JJC8A~z%$hxFsdPz z0*qA~=rdL0RyrWAZ&u^y9JEOuJa@B%WZbBrwc(1Sq6;X*lU*~|a0NAaGUy~wm|Mjn z&~~5?fOY_V7_<{;9P&*C<(DLoN5x5?r-OC}?E#9nP~)=Ss=!;N{h-*-igCr4)22FB zj2()V5!qm@9Ye8vd$!HT8XD|6#covW9>wlgtSNlSaZ*MLI}~eix6G8m-7=G(<9kkm zH|iGVdw4b53V9@gZiV%l!{d9LMZ#m>BXbJuHa(H|$netDvX;o_Lvvu@`ym3G*WLJb zH}2c;ly#r*x_+pQ9jc#d+ zq`laW9V_2J-#b<(C&ahXVnSCHfQ9=GnS0ap9Q|2}ZI^{0O{d?}I_ z|Co*bUlID-zVLPk!(C&?7n&M$z$?nS_=fw|Ln}x7LKYbA3)$eEr4p+|ll;b+X*Bb( z%W&UU4F1OOeYB`>-$%d4v9gvU%AeAamBri} zuuoa+djqIKlgBaqAXO0V2U(?K_(2xv9PB4NIQ#xsLguaQH2JgPkKoI!b3xApJrA@$ z=n&9s(4nC8Im19%?+Zb{0KFLWHyb}}$9tn?F9Octn&+e5l7{(0I@yQ1*MMK>TR}Wbod706FZP_mOre zmU-J4a*8QaWi(xoZBE`g6x4??wZEW)oLeogTMGma_p*Zkoa z5PsJI*CGQ}E8N@_UB;QJn<5%vlOj$-l*6Q8z8txoqqe-0aT0mkERG~W#0__72_|3W z>C7~ku(o{J({RT{t4MsMuaOnKGO{=#B`6wT{VAhc|LYNR4xS0WV#;;seI8zD0OO5GiMXxDy&J(GS~~9!<=V5O^EgEil3_sN2*XB!%c0Q ze3_@?Z2;Q2g{J8x`Tp;=5x55LsTxZAJKG5VtvfDg3Dn*nkW=D$g2yl%(P+J#CQ3(RWHvihz$R^k!<~y;A~S-#(IT zUFACf$}-xcXEcX{EN}qjp?@?7d9c$w1{WG0#9SJf{t*E@GUTp<7$5;3t&%l9_{9sa zy7ww91;RskLhd?<1?t}6(B$?$6Wmm55~!MD&YyKm-Jho)DEep+;-eGZmKjpmFe5wyF?PD ztR()$ui>W+ce<%K{_qf*}!h{a>fLsmknB!V^+2&hE_J9W3k4S=p&>s_X4k z!V^-Kw6E6d&Ci$3a8o^Ir*donUvRMz6aedeAF+^C@LXpvx0WS-&-#WNbS^h?u~uGO zR8p2(l2Uyd>W#U`$a-zLUKbcAv(7en*D8))IQ31Xo{@$6T6MP?(&O z?22jGT%nlZ>r@M0C>DB?232F<9eq^UdctI#>d9J9ncGHcJ!MjLId-brP91$a2=<1% zck|xX@RxhTxL+bk4(<&%mc8NajpW-nG!%>`?tOrapN8K&8mafFWp6kRj^kPIy-LT7 zgCy-3_d3MW?RgI2^W5FXQ<$+^SA*|eo0s|3wwKFUIMGnS4FL)8FPhmVCUz z83>;CbbB}`$FbxaZ}SKv{?sPl)4(T#k8h3UaV+^x#<~9BnP~GJOTI^Nq8vQO)9K+L zdu{Tu8U78P^)}zJ^5cAGD|nuyW5Ypq$uDfrm-aClcI-WlwFf@|d{b;*ns#k_@XYV# zznR}+@I4ZqpL+wKh{XK(-r6^9KGrwQN?VDp_?2p(({?K@bt%7afvau_2CLW^d`l6_I7rOy`Rl^ zE40m2`1ca*Zy&W;PKG2faQD1B7#J@@Jinx*u;@SCO#df4@06_E79_PZN|wF&ML=kp z3BGw>PU*Yi^=bcblF5(}!OkJoAxe%WB?geJ||#Y2jaT+qmp>JE3Eva{Z{*}iH`QC=e{he6qfs+qL?aYT z;Vp-Fc42Z+Ap{P&M{qZ0$GjZk<#ruwy&a0Ren+UcV0g|!La|m)*v^@P^|3>-7C-ft zF4!D9^#9#`9hq~#ZJW7JF1#`;h3ti7*8G)ajY45;fX;obt-09%XY18^mKV>|N>HXQ ztYc8_$?PB@UQkmPLD|+cXFDO@UNEMk_JUSF{55AeA@~8mO8kae19%0b$GHgcd=`GY zc)cP=MQ-gE8pQ>;-+FX@X<4|3bsk;ag3S3H0j^iKb#e=(X^EC+JFM7qlCtpN#A2+3 zuxoBv6_hU?n2_8NnuCqLL)2(wsk1%qfCX&+8jP5Uqu%XZ_|B(OX?6w8lI+gOESw=1?- zv1N)aSF9D<&*aCJG-dQsEKRZHNl3DW7RVlSDSjDYtsV)C2fO{wzi9Q`iTTsbV-4&e zlI-X|b_e+pXK2aJ;gmyLOtwMzf(iy(g0lQ1TYEfjw8xOql))+xH`q%_2!4g1T!5Sc z8pJBz!JizbnYq}5RA~_CP>c!ljw%R}sf3A&C%baMN74`F1 zR@E2!02_UwSQR~m(F;Sd;$;lRmNgiy#bC4+gDsCnP_1ABB4>qZ$7vmE=d-`kPGM2u znEbzM2j;eczp@0hX$6Yrm>`w_imo{_ca_g((!YxBoEEkCrl407aV5W`^>-60~Tgb{IOmS(Mrx4M`yOS zT_@Pe<0ET%2Hl%9d~4c!&fPI~Ey?T{k3t=2mEA#WIc$HKhEtBeq?KkOZk(gpXdQ(3 ziTHUoS}#-<<3C4f-1tup?>zWKxyOHc+n+k-;q?R<_baG(jQtg@Krus*_HTo*v3Q1J z#d{hoA8~`tvO{&OD-@&mGg?SVveO985PZ{ zq~qrr7w^dkcR`#O@QA{%_unwCKYu4eojjL@M$PS;|;J*LJJ@TyyiHQlw$!myB5f3xOBxNx>{uIoxKx*be2}~ZLBfD#v z;V#%|AMe241%`->1S>vP(Mki(Xct5Ve9#HIM#a>nH3&r*- z#vM?TmTQeBKR!!oFj|_ymMOMeF(0mhgQJlZ4yWTW+z~&s^JzlSh<6rDQefoa;Y?{6 z<~poNVM0vNT`>I710C7&{STk_<|;2sc8|Zc`{-GVrJD)(STp7`18-7Rk*cDA>_Ei=0+~%^LOIxd)EUhdn*uYVaokIE-bc%1XwvgW;ZbUs*4 zMm|sDuOm;~QTedO0pH2^ODVbztNZrR`EYqS;NyxddpwVNN9Dsa0Usr2J}>Uw^X$?2 zaBxSxz)&9{9tn~ULym;e7WYtj&`t1&QzXLx(2B!@`)lN=rwG=k@a>7s`najm>{P-NvTN8-QaNS4-D7RM=@%@6gC!5)2`P&U&Td(s zZfVlXO~w0VcnFV#w%78IF-FXN$G+JDYX2pTs2>FGaA`IksrBOuBPNmydHcHA~71@mOnq?xlI9 zxm(rxbk0kf_t3A?A-Me-+572Utw1w=E7Np7Df?~_H?U9x_j?T7rtCl^| zW#D-LzhN9uaIiz99b=DlqRk_W$`|_lY!dKY;Nw@dc~FEj>?wK2H{xI7*B(5H_7scb ztLz1O#%1!=;MWg4ojC~MI2PaMI5!qNXK^6H!E$TM?*rh4;2A(iisM*(+fh&vcm~jU z;RxU-`FZXMBHkT?Q9gL?PG$_A7*0m^0>k)bLa`roM0<5Ie9X2sz7!O6Jf>Ya81s4X z>ZvxqV&pfyE7m(W;lgpO{Kg``rC2v?gzG3CnqSynCFj%Wm>*n(9}oJe~8&jopbpT&sw0r+Q$uh&iUxw*6{t-ad{^! zUH|rzTmE?T@GCRkdvd+pEMDtADc4UDd`It-@)6zI_FH)YRNKDolY$?;Z~F`KPm?ym zWB!=iryx8SzlnBu48oi2c1uNgvmH)Dm>*5%aW;PY@$Fc}ZhGdp`{8;l<9{Bp@U!Z_Henot%_iwJx*jeSuy|11_LM`Jin!#I~P!&#YRU zyod~QUn|aCLro09FK9i0zilP-JpP^}p|$wS=}6Fe2YQ}rPPmln@j^-Yp7DchoKgREzI*GdI4tG?nJIoM}&=5L6fHx?OXE?;@%$PrF(Gkb2I5_NuGo_q~^%Y!W&O`oOS;CV>D31xzq;6Ac;=6cr@&PC^x> zg{ESNvMP2ox~OZx#e%xl1QNi40yfZ95W9%M!dh0s|NA|2@44@VMA_p1*Sw$Potf`F zGjrz5nR45l){pghbg6v7*Zb{LPdk`GnaOKi?=3M$>F7}$(hrursPc*ZsZ7Pvle(-dD z*-}HHa%$ggjchOv$}asBB)_yayL2dr<Z^-h$Rap_A?f(?XDmO)g zIj3}6cKWug(v9gDKyVJHDmO$SJ0y=r@~qOA(zjqrfp|RoePv}-w*NDeX*9A!yQDCE zB~pPY&Hp*rAk$9DlKuf?K`t-nq<_TLP0dDL*(oWMFw0++h0IXPl=MneD#Zu}PNvE7 zzYP9hR|tg!sgKrWr4veGN1BETWeccz+9!xqQ}HBdB4+rrVXml0wtrAcVUB-mcIjjoF`8&}wjZs@MxLDQe}ltiZayYa z+6-8V5l{8kWTQz_v(isb%1MvQNk2D`l|CstCw)?UR{FV}a?Pfwlro0c8fcpe!*6nwqmbI%j!7AZz(}(DCtE%L_VXEiX>XSzg>d zXZf6roaKeRIRMN)ld42;A4|&4TT+opwNK7lvN5xig9m4rT#*2(AqjJj@ z+Oj)LjYyWgd~n+G{%DjHGEVElX;Bpi^5MfS+Q>U#X8fOs&eZn_9<%abC02f{i!*y#Tt6y?C(&pgbvaXBRZ1b(a ztmHExl{s?3la+inD=fg(5nHT95QCVFt44xPhdt_Mo-|}WhP15vBHnfK^j@#%BOkw? zi?n@&=hld>kf+pc=?i^bgZY%;JT-t@R}mi}A8z_lpt9W_F%{{O>{Zr!yU3Rs(M$5> zPA5jLErC^5l6AlJou41oQ>V6N-iu-0>^7$`YwPxi9mw0=X$ANDaNHeHZs%Y#flyZ- z-Ef1<-P~kO*KSf;bMU?`68**B&AQQdk-w8@usO)G0{7Sywgo~PC6iQN9xgKsk6ZaiCEkJx5;2+} z-Y>a3(xlRwAQWFDFa=CZ6RB|d&TaE?90(kSO%>RqAYE4S1QhbP6dWa)CL_~gDL6tb zC>tqwYZx~`1FfOSkc_*FT8qH^s4zSf=xz`3Q~h%g_emyi24ce!!CMupM`^~3sF$G? zM+!C`37(Qf6Mfh&UZfe%_LeH1h+6bZPC+`i{eQxX7B|^5?a&rTlg%TT`z}fKvM<>x zx3}1N)>*zjk|Q74eJST|6GgLAy(*=?1B;DT9XyO4NBg9XxUG$mf|k@TIfyf^6rWjZ zU`dW-l5f1v)6twXsT!61tSH&?728H{Q;12ymhT+gRW^-bThB_t#_Prut$tF7;^*|# z7A=($|KJN)f3>&T(?v(AA<|Jt8|=k+#GReo3Am@LiSS#;B3b`W@XzKW~5oH)e*=)aKU*;$d-*E}>%Dyh{hZW57_*6U=NZ$hQY4Kxmfvr2EN^6;F2f!RJ6=jGKw}XT zak(u9j>p*2;R=Y-S=N;i&xBXi6v(Q*Bn-xGh9LZ^#A71*ge92MUB``c#3*K9 z)0wt*s@>E2%w7yBvXUo4h%ZEt`|Kbb4=u1#rTHf6sBZMdLD=mpKUy7pu8HrKv}1go z!;LLNR7FXvL^31kAcVI_JSieOT+&{mJGSB+(f4#{?n1#Pp@c0`N=sk1(*~1t0Osi_MC4fPZ)ZJ? zzS355DpE~#2^m2qb`{|{LYhAOMPQ_To=ir`DH5o)}*i?&i~8nf2?E^VX-7;5m5++cO6H_~os|7b zU=pTuDsvvf}o5#s$GjC~OU>uARm%M^RaV7(hy zf;Z$k*^AL#%yesppa=a^t)8OmOmOWZtF=GSCM>r(UCk`PE)gr$P7wJxGPafY0?CzX z;dk&@obh8sQKL86Z3%F)dme9+2JF?aJ~Qr`Nq-mKfO2P;iQaPUBgs5c%!4b%E10+S zcHmt5A`!V5vF`=j6KDdF8-+N1;?tD)z_J87gQ^FQX#f}JQT6cK_46;q-iP(%AHDSB z2JrV9z~5{Df29F@Qv>*O4d9R0iwBne&;Wk00sO0aasN^*_|8t|JJ~@o0BMxaA7>-1 z1HiZeAYNdjHR0d}fL%}S1|&H-6KQ#Mg)dxibtUg?194xKWt=k}=SF@5!@c~5AbtXJ zImb3vCd_3W;%2Wa4qE)|g{vzbkx6_hk#LCZid1zin# z8R&Y@rJ%elSq926^>R=yaA5l->|9rZa;94j$`R!%P%ijNLHRUr1t`1r)u627HJ~Sg zt^~ai^pBw2An#hx6!b3)xg{rpUJrU7=q;eUTDcRHtFtwrn?dge<)RAXV2K|C?jxX4 zpzA^7Krt|t@X`naQpq|{y#8DAIOs;u=Rm7KH-T1zz5==l^i9yspzncVGA{WT6t}HQ zz68ZB{gSUhD?kr|R)YQjx&ibM=!>8>);=$RMu5Hy8Uwl&GyxP30!orWw}CbXeGimh z|7{1o5p)M=Ip}MkTr<51`V{C}pqwangLXoDw!^=9C7PjFt5C7gPAHab>+o@*X!w>Y zh8ANVpA1ctk~s>U8~p)iruByJ&J8qY_no-DfW(H-z)Z$V)5ul zrfx}2DAo!p#$IXak){~kM;Pop#V%BAx?-~xTdWv7t~hCLQ0x}P?osT1#n}Iid`~I% zqGGQo_K9M=v0~)=hhm2n!+W$&evO?FoY*UNf?{}c*5S)kY>;9@6kDv=<%(Ua*bR!^ zquBk5ZC30h#da(9fnqg^eWRE^!j&(|3B_9N6lhxR*@`Vz>~h7fRqO`E?o#X?#kMH+vSNQzY>#4JD0V=xpA_R4ORRaU)x-(K zSuGSxRjiX@y%oz;Y?xvr6q};hG{vq~>{`X{SL`9hHYm19u^o!Nso2MgeXdwkwA+?( zPAJxDuGq1PWhmBLv3$h}6f0D0v|`1I%~b4K#cojSF2(Lq>=DHtSL}7gb}IITVh0q9 ziE-ZcbylpaVxtusuh<;L<|(#9v6YJ5r`TG>o>i<;vE7RC)nQ{-HHv+s zSX8WAk2oiUHd5?Z#WED@t=J&NhA1{#vGIysuGm$I-LBYOiao8^^NPKs*sF@Yqu6f6 z4k^a3SWJDJI-xkLrDE+B>#W#Wik+j_MT!kqY?5L{iY-)ZiDI`Zc86k5D)x+GTNHa) zvA-#{N3k35GERFL6K`=ss4Si#nQ@X&lT3YEDAr0bOd9gcX+KsuJ?cDjs=>LI_-e!F#Yjf~JDjf{p_{42o+(wz@c4CVq#5Kj7f&cpB$7g8D$I z{QxNKDFT$&El`GY-NH~TuVWnySF#d{<@U)AHc7F|6}w8Y8x+IUp_BFz#c<{4U>g+M zq!_NdoHF(*_N8K9D@MobMrJx*H!{0@=#N0^KDXuL3h15JkJf7p$WgZsxnCcJJ{je$ zwKLH-%+XT&274s0gL(cesJB|;O{5$09t8GbS6(@VJA33Eh`gE!Q0~ONv_HFt`(47$ zu6ATmbw6CbLwMQMTweE%nQcwry^#9%U@wPUp&v54b?+eZ7HGOOiSx&NVgF}@>hH^; z&ez$OBmMpbaUGrh93Ox?=d+VL*6qn{BYpn*oUSLwN5hrr*~#t>HKE-&v;(PhxJQ$W z3Z0T%2&lep%$r14;+%Wa=%4Vm&bv2o9AUn%rUKmWt9er?h;{EIr#lfe64f-VJFTW*9Ckug|49N1KmK` z?`etVeg#9ZykF&DBb`vJHC3_cictoWcCljY?FPGDG0J4HhZK8CvF8+{E({+ZBr+5$ z8=4qwzhW4#9KIMQ6l*nAtfgW-6+1<-3l+OqG2Ur6WlUG>&psD>&GT>h* z{TEB?c5&?@1MvZQq;*ON9~am0(J7HJ@zISVqy6!5^-ax9`5xL`J zw>5)JP;815GA>C>+WCrcgfQ66iruZ)eTr2nwnZ^sQ<$_n6+58Vw~C=7IDEJgkr3|7 zDAq;lF3FxiwlMKM0NGT4(&2(Cd|N1zQ;tU$59z~}TE%9n)y-{aqp ze{VI$a69~)tzJwh8gX_&xT{EywbH*0S|@(k#ybA>wpQb+6Rhab?Qzb_fkTZ+$7^d7 z3L<23n1>LbF1d#}A3)TESXMXud#*8uwi7vi7!c)DZ|WTa%&7B42II*~lCxIW)owLLZ^2 zB5}rVv+bB_R*R>bc&U0hskl*0c&aE##U`K}!;bwJLyB8go|7u-h*Z?R-)8$T)th}@ zSnZ{{$VnA_M5-7`MQdl@J?rwuB_>tqmO-(?!wn0#Ndb=-sZS_R98kZ_%?eOOIh^f@ zLbfwt&+y|lA+wnPnVhYMLU^QS*u^3F7}px1W&KX_;RXfAF%uk7A`e5;`QseMP8PUP z)ZHLu4iY2UU}Vh^JnO>{eEvBDuLe2?^Ko9Jka2=K4Btp65s2sNzIL{Vr`ufUEUaDB9y^`f@=%zi5h-g`- zT-@h8FI5L8mGFdZ+9P9yZAKk@#WeNL{k&A}T^=)@gqB5%@XFe~Y}=P!s)xtX?f86?I6K;hRnkc-H%#lS+8P$~q>jtX6FgR(h#; z<-kLD!ph=W%eZDC%Q0W~c+B)TawnV4AWQ%74V^gGdi6Sa%D|Rhsx&8+@Gu+eYzJZ3 zXB8sWz+MLR?4RRoSkC+JPEha1d#Bvnigh?#csRFI7(QYoytueq7NNUdHfDN})xdWp z|Fqkd2J&&YEpqph2>*J;ap%&V6o1%nqRx49yA$y2T6 z?#)z3w{MSszyOt=_sR`sW=d$v9wG2PO7_6T~%FYSNStyhn(HZ~{f~xVZs$Y(Q#oGaU@ypQVE?0Y93Ej*z9aWP+6=x= z99~Z1nGNiJl!#gKAZJP(6t%wnin!=F5i?~o&NDa~;@gXJx!~!*5eWy&tuH?=ECzz7 zBj*wv4e^yAuXDlkT2Ru_mMA0L+%hlk?T!QsrU;rzI~IJwZWZXG3T4eozLwQU}ay5OG&2ik`A z7kVE&`yD<`*7faw)E#)-7;Gnue;&$L&;Cclz%!vPU!DDm(vZg$;CZ4hUp@J~0iO5k z^3}dEhX@Qka=SC4$l!1Kqte0B1pef$MH8y&u%Ymcqq z*>x28eFvU6-kZenbNMv|PaB6%+pUg#)Zb~~Ip-+yy8t|6jv~J$;JNN7@>>m_`|9R5 zJ>=R5vWY!AJghH|%%pnQm?IZD3o>fyuC{jcmF^$^4^{&jo`>3lxAX!6K0 zBW87;f$b90|LvWm%o_|R!oD((W-lqHaBS6x!hhXerMNxRM%cyh;ZwSH%iu?8;qfE< z0@bs{R6WqYz3bGsXycvuXO~Q}cl12gzUjG3^VTgp>BlF|z5y4l9J!P1m#%oB#oP^N z#Et9MukrQM-`j>RFZhu@d+>A+{73FZ#jB0__M+k&w@LPA1y?0MG-j2(nn*=}7=g?lk>)*SI(*Y0eU6l)ZF8+B- znb#HV@XyCc40ppnpAa+L8~@+nzbD5aY|Hd-_^}CF$`5rdf2d=8LuaKK=R3b%32u?Y z!#mr5z_pRPwbTDEmp}i}{jq*Mh0p5xYd7hv?*9poO}JwD^*mFVv6y$%BC*HUF7&IH z5$1icIDX?Gq5nD`OAz1}4_+Pow#S#d$5$b=@!K9>KeGv%QP6`>Q>v)ydHiyE3GmY% zD@8(le0`#X{%`E@RVa1kD+WdrFCb(z!9|DB1ok@n-@C_GA*vbRw`Yz5DiJaY;23BW zAbWDn#x*}S`uY^>nDC9hEF!vD7j{3(KXr@l*TpSu_JD!=Z>&pGnywXP;t}JC0Q#+Q0ODZZyW{r~S*;Rp1O{e=sEtappRq zcV6We(RmZ9EbL}BHnF$@A{W2FpB?;phd&?kXTO#2-vat~Mh;tJbA%VaV8$=F@e6kR zf*-&8S#^U@e6?!DHH~Q>r>d22%srUU`@&a6L?TIMZ>AbWKSUk zo=FKpxsV(g6*{36Dg54z*!3xyPUyB2e$hs3O-i{FdN5@jLn}+O(r}{a`_cj3$0inS zLau{Sf<(T}t=t;~^igiyHlWyP62I7kZERLz(SG1^yDP7BGqW9(Qam;{aq0$e%uC4} zn>e{5cgcpp*u>Yj@a)o*a@GKOV&h4qM0T(9L4n1rej@&$;Ycj{dTipzuZVoWpEt)Q zPTanH00kVm1L3c8D?g`fP)-SxgJq})i@KJuHaxS^oPnrmh(DIH5+{}q=)Mm&c6>H@ zim=%kOMVkSwQnJlnjQGbkau@Pq_=tET`+D$pav=_LR(KifDEQ@hkfp2rHWp_Pwk61 z_xjkxT5Ma0R<79|l22j{>XJd5k{$Bqr@QvC)WH(Df);M$o}s}C5{l^AmMHi zqL@OUrVt{r=zx(Sar&DrQCq}B0x^L4G_i{^P-kdZl>QI=)Iuq4sfjh8GtyZ^NG$q9 zYy?eH`vEeBmTR5HOe{hjrF=&Rq5@X~$hZRh^)$r$EPgM}NqcUm2w#L0xR9-#2z{gW z`3$PAYhDDFAuj=lDItHO`l4OPWcoJIKfv0*AraevmH>OA@{W`wu?VWGMGq(vlY*e= zGMOPIKr6JfBDDcAJD4dgI>h>z4Gpj8ZONh-QD{jNv&U_i*V!AvMmbQELWJgtK{wP` zo~^va%wES&4XiNKN}&tWX`rhwA*Xbe-Q<1`KiYal6>f8REd?=y z6k?-AP@+^Ad&)^ok;MTm0Q~@6@j&j9Pwm8IPh+1qWR3CDXl9KWu_0d#V0R!F%QmB> z82VCYILB~-tepWFX<+jcKf|E63zQA|62jZQK|OsKMKQe8pbAso2Cy)is2?m_fxyBi zzl^gx5ntm>2eqS93cpEQcwR9D#CB;uz7cg3Zs6dB^mw3G;Oiy_`c zNs#XkWOM~Sn@A23XD&;um9f0$Q%X5CiB|9_e_(XAj)6i#&PkAtQi(R9RcdqM9+8lp z3`IM$A6rlQOSp=2gCcbhDIFynsVQcVV#>Vnp=p)cPr&BPzz`lhXOky~saMFX13qZD z_7womlntS)r4nR%(9Bs7T}FPVvWRl3z%XIdBz2;7IQ3!uoI0iDE!tq!sRr^jO9&Q| z3yunMNVO2zXxPJRha_f=!D^=E8Yb2j;muA0cQqmwbv90Hl5A03gsUAU*Y+K|V*ilJ zqF`Z{I+HhV$rtEIi@!U3I6E=bEQSzpmq)1&%7To0C~~2ekG&JQcH^d$vZMpfl^*v{ zo-{|(N>VA(=CAc?;h1o=Bc z=Hd5gS3CIg4S&8EyAncUAm$uUgE>3HVkiFyz{jARusGs}4KwVqMccvg8qR)>Trwktyp!+!^pDdruVq`}Xu)rKR zbaNmn-h&)#4oHy)0As$Y#loWaOB7vv07X(z%z7f;^siXjj+Q{Sn}Mwn7QHkS^kx_# zl~Bu}c^d;?$O)Ri$aO%XX*kP1$+4=2-E1NI&%y%s{e{I69-me^Agwg3`;razEZ^aZ zyd@QxSkL7x*^s$Bt9#y(eK3Pb*zZ&>TIj+_TBn|-Ni;2s@{2g%o3>?4+DfKN42jPA zuHN?AgKGZ|YA-fbaRu6}zV^ynQUk@Yz0#$bXtk!nsEOJyN?$}*Mz3MdmrDv0z#L=( z808T_#|b7b=5i*hEbDOosGYfGOC?R7f8$wYty<)tT5(B&i(y~q8RO3?3mii**w9lG|}7_Z0r7Ut&?9QQoMHODZ`Wu4qCjvX*Sbh{wUBXlX#i z>X*1M-j_RSJIyDa_OpdX6Oa9)Uz`3+?x@PVQ!8im$t}BH1j_YqbLGhI7!s^VJl^7- zsf`X-cJpOj8A!@48)%!v(?5AClR}kYP=WxTB8)4qRQK{U)xBU%buVF4F_XXmIUcPP zYOZ?oG3u;CoO|W&0Yx6>%9h))T0G(IW51}1m)je0Ytkyje*hjx+`U-2`v#wC2CP3L zG)h9J`i&2cfVBjn$%3s^EGFP$%@CR(XICh8vtqFr`HbF zlHY;|Hxk09o7gV!G;5;u7knIdzKDH0(0h5$+Fc#7!e zUNn8eII~+Hv%i{mfnc?!I8UlX>yhX_F}Zei?C)h4LeBS(N^Q0#jI$%<3ra0cz96{` z9CyOPXhqadIYOnP*Tr~)wOFuPU=PA>xffJ2N)4;C*?yHmohNl(VJG6#-Wk~DsfA64 z$yv#F;M^xt=P`JtIw?H4*%*~9>!i+?;pG*xU1tg84z(b!)H%bQjZ9hUos#S<*-Dt% zj1{9hKKe0K_9S17bqD?*@imHI10KTpwpR!JU1U#68rDdu!Yd*&RWU@A z9xwe0Uu9coN(sM<+-7wR#7A(i->hUm@CwQIOn)3+fTB8XS4oeNxKH~`&kzfmGu&*s z$wtVKdj3whDd`R=xwn)|dwx(#SQ9bBdeXPTZ^=H)Yz($BzldR}l#qMySm2>#Yy_j^ zUiRlGmwN%H;jF88jw+Y90G z9W7eA?Pt<$Z`0}(B+Rt>M>mO?W>>^^6m!@H-fB_J(+PuN4&nKkO?=tFYb3w^lHVO6 z`Sr9)qmPRjXTKQRM)JccYmekto^T5Ct7ETbTwyPNMHF;`O?$1?3=hSxwo;;Nqpz@U zjvXi&W+JmBqu_+tFngUj{A95!+T-_P6@P(3E)MVXX7@;HgqxjkD>!}&Y6hk`!_EH5 zG^?{k=9&1?;w;%6lUp75A`R%vDvgy&k>{Fw%y zA);qT?8UCUG`(B2s%A52zZ!g;v~MQzaNGA>Yjv2krgbUVJ}LDCF?vciM5O;i#7^r) z`&j>wX=B_}hb%EX9zf?MdXB{&o&cVZ2;oBln5b1Ix-K)MgAWx|N z%wC<;-)X|p7n6HdG9@gO?lnBX0c5n*+rHlpSc%xZxs$JvbV)W^tw?vVlv<8Bw|3=b zchvWt2vtcQz2W)TY%S0JVRVxyk`;o-)fM|fSoid1FzyFsSH z9L#i0`7V<(AD2#q-fBH9`a3OhxV6w)?Vm07;K^Mqaa+4}ir7av^0`^k#oCLG(1qzn zN`dAy9J%wq;qa?eX z_^@Yg9rc@CZK;w)Vv@x&vfrxeFG%QXEs4&F$+Qo~{#}Z!0PizWWL3h&Vd|eVyoW9v z{w4~&Ui$rbtDSu#`n~NtU6f0OFBavt5M@%rfufXla1F81w+f34j*)DHaZ=I?Vo2qP zXN#$|(XoIF2Qx}N~J( zhjn6G`$>B@+zOoQHwMUo#?-T^8j@SD3D_9b8;xgTHe;#O@^b0G`J%l;hTrbAuTnKmq}r{m~{=?<`T(e5w<}}l~in(>!lqolP>QL15e1%FdvxNL*8BXXNXO4 zcIYo<-G!laC^jN^#Fyl-32v)cF`@I_c(_%F5w2h2kBPUzX_)pTHX zXruS%#riXmeu<=y^WO+ry$**<1Y=F?Ox>h2og^Yqg~rwLTxo8K^sr>{ROAF)TyDoS z-PrfM9T|wRF2g^ofl{%!0HbxgQw>*)ublI)n2QiMtJHf%CAXrBg^sUAg)DQXnB07^ zaIc!p6Bb#S(XT~6Vy}(uD;6F^xnE1!S0|)~jVY+2+3cKcd5X05d%m{TN+dJ(_7@SM z)Sdue9q!QILTv7mz@ye8nL-aE?Fz|$ja0?ENZEii#$k(V*+;Hyjf$z#Q$@v(i$!o0 zaMQH4?ubr|nQcE3+e@l*0K8^L?)wr>4;!y$;&Q@V&br&Q&+{i>RX)pF?>o)!tr(vc zdwMp^GR?S2Jv0(Mn4ZC=r&Ahy#}fIjH8_GgVDIPVq_n&Fa(!-e^@j8+bGgHQSYt*^ zS+jZLZ(q?#0nWSiqfWfuK4-X*laj2K96O2TS?YXgjWxbm))SH|M}%P_-D{$G_Tx_^ z%^5Nd)983_9iR0D23kAq_x*E(&s(6}BykRprcVr$KCu!hB1HNV?52>d&KPSh9+1ZB zBo??@gyGC+s#GdfT!-@~N#O$%z6h)0OiU~66Rc9KG{wUq3YdxA6N)aPBsZW>6Lknv|F3dlU|txa&mF#Nw3Utaxx|Kq*rElIoUv&%cOfBADYrD^CV|@ z4QUmx%tARCCaqT{zTL&qF|@UZn=y@==Z1>#5sx|LG&QRIAy{a?GkLP#9TW~TaMMVy zka$omjFvD+;>H^RN3PQ(J{x`ITx=O0S$90vhRqCReFXIpTQ|pKQ0nw&lD-C3hPH>k?>hO}0Duw#zNX zS>TnmL*{Ax4N|OW_BOj?m^5Y-V}qS7xs0>(WwCB<8{Hz< zT*-;IU0hw9gaz{3cAJo#uV9$2+k!Cw^IuJ6AZ8@U6D1 zUEUtyS!(KTN$r{X$rATW9rrkKc&5Hr+OQWE3FlfHp^4C3&2^@9AmJ*tgO=H{i)MYc zPoU)TEFhMkAXYWdOeI+~EnaQqGSRFkZYc2Bryp6RJp1L;HH_XNkNU0s+ z;b*q>XJ~X6YP+wFW<6W_QMlXq3PNn@Oetk97W@}t!yj+YP(HY-=WsR~GhmtP%$Q0NNJxOi<=?7ATi2LHx&~NW^hh$SPrv zIS%wspvQwg2pR-^7_<%OFnQ5T(x-%X0ThJMx^w5M} zT=E?#hS8FtSa4qgIv?tu13CuFiJL(A%moufNefJYcY=NadOs+5tdsCR5AuA4xf7&> z;|t8BBo!1LtK=k5sJNs%D4L+802E_%$q>*cphH2Mf{p;CE+>GZJxZp5wgTnE+8T5T zD0N#38U(!tv@Pf=Q0nkrQ0ne6(2k%RK&i7$ppdcTMbIvwuYjh3z6OeKtV-SjO$Xf# zdNL>{?F`VbKu-bv0kk(L21>pLSrQF;8fX$|CTLU8exPkZ&j3vW{R3!U(6d3$2R#RL zENB7frJw^q7k~}~T?%?G=vAPDKv#ke0lgXY0?=DQF9cl;S_rxZbOfk{12$siL+P9n zsr%gR%{hL7RcL1Z5kv0X-I!vjE#59hCNc zDk$6FG|)DnIiM+^=Y!IIM}u|%oetU&bT(*b(91yC1`9yh21`J@f?fg2Hn;{9PrFMv z!>|qR1nmWSALyx|kASibo&@a+S^>&7s0PgheE~EJbO&e-=pInE#h0Meb$2L%x_%jy z@dHjg21*%>IP8u3qb{uRz}QZm5Qbe@|BCmL+ca&t6N~+OP!c?`u`X^M4MELSmFis9pJa+7w3Vx@{*qu4`=J)#)zC!4f86yq)@2HUIH zkBZeQ*2?Gd1)WeVZs)rgM>8W|refKOjZ$o!Vha^pqF8<2M&Q25k?$$x+oc#sNmK4; zitSe{24khETY?jcwc04wPO+|v@wS-BFH5l;#Re!gSh3lPU8dOOie07HU5eeKSe0U1 z6#G!IPZi^sZ|d>AVu65*MLVHbtCeCw#ppTFr0uR)reeIYYU+EjVj~rss@QbJ7AUq@ zG1u=%oOO$0^?4w9T+=?Kd@m|S-$+IldOtEU)Blmd@EF;_=SFvD7ICxU5dS{*nY*nQtXgoHdYu$z62)} zXC*7vMzMB^byci~Vp)piC^ky5af;1Qj5nxFxuuF-qu8yA-J#f{ianv&M#VNO_L^dE zDRxk?9~6tg?Gs86XT><7SUfjyG2SOK^7T@zk79X>^;c||Vj~orqS!RWRw{PAV&#gh zQEZ)J*lNm=rAo0ain)GA;w;{0GBSUve6BB&SnGSmep0?xSWg-q1f3A(1;zNp&E%J_ z7;haJtXMJLr8ZcJVoMb(S8R=98x+G9YfjoX6?@F_LE}KSo0eB z;+;^e)k3jWiuG13Q?Y!-xY9Ru8?V@8#kh_)WpE{L`2M2Uql!JR*ha6q}&f6veonH}#mW*mA|n6njXqM-=0F-K5>1 z*xQP|r`RFIY^-Qa-4dJ-#!1C^r@-XL6};i=tXNmY1}HXIu@Q=mQH<+!lOI>=Ccot%|*?*!zlor5HEBGG#Qz`rM>#?u25kE{dfq)?cv!iVah2gkqN} zHea#jij^sLvtp|i`>SHlD)xe6TNT@-*t?4Tq!=&1Ox@z0P@I*dSS!VXie)O6tyqC# zgA^-NY_wv^MI1ZStfyjw6dR)0bj4;XcDZ6#DR!@74=7fx z*b9p7RqSKM4lCxzEf1sTR!%6+3M$r1u|A5Or`Uyx%~0$T#a1eIy<%$>dswj-730o6 zMwU+$+oxCnHzJIlqn!}0ITdTCSYO5ZDRz-!!xfvO*gVBnDR#4Bk1FLhJrbM{b{to%wPIZr>!H~Bie04GIK?I@c8Owh6}v*QQpIjm>{i7d zQ0y;??NIDZ#r7)pv0@3hVP@)!>gDK~}5QdNguEu~s+5dMb9FVizj5RIw`+yGgOz6nh(?_A+L_=Y(Rd z{A1j5`GpNN8EcJJY`kK)aq^oNzTdp?!3xAHk31*CrHFjFMQl9oljC`SJab9~Ekb-8 zXffzzpwmDXI`|z9{(yt81D%fZ8$oA)?f{(y`a0-r&`zLOd||DT=7fIp!q+#M4?>T0 zXxS1^dCV(G=;Ao8cjBJAJI`%S-7ndi3GYDMqsKIXquhymY59`H2{;1}ev`*cmD7AT z)*Qci;rn0cg%1-Hn%VQ(vI_l?mUS~|TTr}ZAzt{@w*(p`%~;^?n}v#%1r7j!u2Q=o;QFMy5&eF=0FDElUE z(m8#Tq2Ijl)iWLc=7sMZ3|m~bnd6t7;{N7^?;LxQybTwG!s__R{TdM0%I>ij0rq*^ zEK)B=^Uyc(zK*pRGzE0I(^r>)p6tX=17-YlCq5gr6!E!Ed?Dynh~r=Il@5NB6Tj7o zbNx=fzdHDHpeqpH3VJo@tDx6_a;%_?zj@)K4t~K4Ut1sEo5cvt!&lcs8{UA@qa8xV zLz@ZV5voN^!z13gjjVIGq*!M@+0M%P^Y5*Un@+U4T$yV9eolMqxbYpVq;Jo&qPt_% z$KmG4`N8DFJWPn)*5pX8Px0>_^_ATXCsFo>Bx}R>ZLGh3+SV%Ad4iSs;rUjNsv*|z zXI@}6AAO+}J?J7S)6I=GYH|ygz~n|QzD?*J^_81}lg0~RsFb{4VYs6;6pxU}mwB39 zIRN3PFWe+JsoAGX?u)FNHPW zbB`nhOg<*WdmZ!bO0-DTvzvv)$wym}Zrxl-dn$8v6T z5YIc|Lr3sDABNQHoe(d~n{fuOpr92)z1?x}@w~$2u3V0tDk7X5-*Tw%gn6Ko(ZJ^T z$bQ!GoO^eAss8SyG9yN4S$G(TvbsB|`VT+a#9ITSXp@dgj8->EBdDH@>0!mQVCC3S$MGH=#XV~ zJ7d=hFVziBD&Yw$i&x}cSsh}Yf6hzwgp*2m!pg#fjF7T6-+f?&mujbzN_fJ`Y7tge z@$rKTy;MIqse~u2EZ(;X_k`&APc*yuz^gKO!pdUr^y(Gvb+NNEAPP@dS;rnxmghgw zZ0}(llZThpDy*zeLH^N=UuJy)ItqlqEc2 zWgQ<@R=C&2JDpU*6Q;vpSXtp-7b~1p!UI0bde1=^hNs#P=S*#YnBL8Ew_EqW?9EC1 ze1->k!c(;c;-xBS+XcghTSRl3I1ciJr#beVtU~2~WlKoR_N8gl9rh zjd4Hq`S6W8e$g~k@J<__Tot)M)jrpdhDchJd!#KiW?CCIeGmKVj zmu_88?i%!tU-G=!y=%9VPZoB1-fSbDH~ZgUVRt5T)aQA#x4RB;c&vInZyvxRdlL9A z;+&0x!&GJio;Qme9${3zBRy|&gUKjfH{l52KeGYPoBYJ7DW7%Vps4kE%H+k=AHdU- zYbYEI@$JPqem-<2XC)jgx4!�iO<@v$#&e(GXuJ%;qz|b4Po_$APy#&zs-l`u0!< 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%0AR<5T&_fP7{J?<86%sh?FbE!a;GsvT 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%0D~q?^NSe7~ovS5WotQM(7+M{K?RuHNo&8*8bz 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

HC5pr}X7UD`Sb;9O0-Rk8>IkaC> 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∋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

tKE(Reb0#C+T-v}XeYrA*DddfpvD=}8 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%0D6Sli;f_!Ejc=Zm|zp--NtH6 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

nLj7R^^N^D9a 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

gK*pKnPIy3|k(%BEL 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