From 0cb6560e5182d47ec53d92119db76ca97fe1d463 Mon Sep 17 00:00:00 2001 From: Radkesvat <134321679+radkesvat@users.noreply.github.com> Date: Mon, 1 Jul 2024 13:55:12 +0000 Subject: [PATCH] [WIP] 35% wireguard --- tunnels/shared/wireguard/defs.h | 54 +- tunnels/shared/wireguard/lwip_types.h | 64 ++ tunnels/shared/wireguard/wireguard.c | 1098 ++++++++++++++++++++++++ tunnels/shared/wireguard/wireguard.h | 55 ++ tunnels/shared/wireguard/wireguardif.h | 108 +++ 5 files changed, 1352 insertions(+), 27 deletions(-) create mode 100644 tunnels/shared/wireguard/lwip_types.h create mode 100644 tunnels/shared/wireguard/wireguard.c create mode 100644 tunnels/shared/wireguard/wireguard.h create mode 100644 tunnels/shared/wireguard/wireguardif.h diff --git a/tunnels/shared/wireguard/defs.h b/tunnels/shared/wireguard/defs.h index aed5ca48..0e701447 100644 --- a/tunnels/shared/wireguard/defs.h +++ b/tunnels/shared/wireguard/defs.h @@ -9,13 +9,7 @@ their license files are placed next to this file */ -#include -#include - -struct ip4_addr { - uint32_t addr; -}; -typedef struct ip4_addr ip4_addr_t; +#include "lwip_types.h" enum wg_general_limits { @@ -54,10 +48,10 @@ enum wg_timing_consts enum wg_message_consts { kWgMsgInvalid = 0, - kWgMsgInitHandshake = 0, - kWgMsgReplyHandshake = 0, - kWgMsgReplyCookie = 0, - kWgMsgTransportData = 0 + kWgMsgInitHandshake = 1, + kWgMsgReplyHandshake = 2, + kWgMsgReplyCookie = 3, + kWgMsgTransportData = 4 }; typedef struct wireguard_keypair_s @@ -92,13 +86,15 @@ typedef struct wireguard_handshake_s } wireguard_handshake_t; -typedef struct wireguard_allowed_ip_s { - bool valid; - ip_addr_t ip; - ip_addr_t mask; +typedef struct wireguard_allowed_ip_s +{ + bool valid; + ip_addr_t ip; + ip_addr_t mask; + } wireguard_allowed_ip_t; -struct wireguard_peer +struct wireguard_peer_s { bool valid; // Is this peer initialised? bool active; // Should we be actively trying to connect? @@ -112,7 +108,7 @@ struct wireguard_peer // keep-alive interval in seconds, 0 is disable uint16_t keepalive_interval; - struct wireguard_allowed_ip allowed_source_ips[kWgMaxSrcIPs]; + struct wireguard_allowed_ip_s allowed_source_ips[kWgMaxSrcIPs]; uint8_t public_key[kWgPublicKeyLen]; uint8_t preshared_key[kWgSessionKeyLen]; @@ -121,15 +117,15 @@ struct wireguard_peer uint8_t public_key_dh[kWgPublicKeyLen]; // Session keypairs - struct wireguard_keypair curr_keypair; - struct wireguard_keypair prev_keypair; - struct wireguard_keypair next_keypair; + struct wireguard_keypair_s curr_keypair; + struct wireguard_keypair_s prev_keypair; + struct wireguard_keypair_s next_keypair; // 5.1 Silence is a Virtue: The responder keeps track of the greatest timestamp received per peer uint8_t greatest_timestamp[kWgTai64Len]; // The active handshake that is happening - struct wireguard_handshake handshake; + struct wireguard_handshake_s handshake; // Decrypted cookie from the responder uint32_t cookie_millis; @@ -173,16 +169,19 @@ struct wireguard_device uint8_t label_mac1_key[kWgSessionKeyLen]; // List of peers associated with this device - struct wireguard_peer peers[kWgMaxPeers]; + struct wireguard_peer_s peers[kWgMaxPeers]; bool valid; }; -#define MESSAGE_INVALID 0 -#define MESSAGE_HANDSHAKE_INITIATION 1 -#define MESSAGE_HANDSHAKE_RESPONSE 2 -#define MESSAGE_COOKIE_REPLY 3 -#define MESSAGE_TRANSPORT_DATA 4 +enum wireguard_message_constants +{ + kMessageInvalid = 0, + kMessageHandshakeInitiation = 1, + kMessageHandshakeResponse = 2, + kMessageCookieReply = 3, + kMessageTransportData = 4 +}; // 5.4.2 First Message: Initiator to Responder struct message_handshake_initiation @@ -223,6 +222,7 @@ struct message_cookie_reply // 5.4.6 Subsequent Messages: Transport Data Messages struct message_transport_data { + uint8_t type; uint8_t reserved[3]; uint32_t receiver; diff --git a/tunnels/shared/wireguard/lwip_types.h b/tunnels/shared/wireguard/lwip_types.h new file mode 100644 index 00000000..aff2013e --- /dev/null +++ b/tunnels/shared/wireguard/lwip_types.h @@ -0,0 +1,64 @@ +#pragma once +#include +#include + +/* + Most of the code is taken and renamed, from the awesome projects wireguard-lwip and lwip + + Author of lwip: Adam Dunkels https://github.com/smartalock/wireguard-lwip + Author of wireguard-lwip: Daniel Hope https://github.com/lwip-tcpip/lwip + + their license files are placed next to this file +*/ + +struct ip4_addr +{ + uint32_t addr; +}; + +typedef struct ip4_addr ip4_addr_t; + +/** This is the aligned version of ip6_addr_t, + used as local variable, on the stack, etc. */ +struct ip6_addr +{ + uint32_t addr[4]; +#if LWIP_IPV6_SCOPES + uint8_t zone; +#endif /* LWIP_IPV6_SCOPES */ +}; + +/** IPv6 address */ +typedef struct ip6_addr ip6_addr_t; + +/** @ingroup ipaddr + * IP address types for use in ip_addr_t.type member. + * @see tcp_new_ip_type(), udp_new_ip_type(), raw_new_ip_type(). + */ + +enum lwip_ip_addr_type +{ + /** IPv4 */ + kIpaddrTypeV4 = 0U, + /** IPv6 */ + kIpaddrTypeV6 = 6U, + /** IPv4+IPv6 ("dual-stack") */ + kIpaddrTypeAny = 46U +}; + +/** + * @ingroup ipaddr + * A union struct for both IP version's addresses. + * ATTENTION: watch out for its size when adding IPv6 address scope! + */ +typedef struct ip_addr +{ + union { + ip6_addr_t ip6; + ip4_addr_t ip4; + } u_addr; + /** @ref lwip_ip_addr_type */ + uint8_t type; +} ip_addr_t; + +typedef unsigned char err_t; diff --git a/tunnels/shared/wireguard/wireguard.c b/tunnels/shared/wireguard/wireguard.c new file mode 100644 index 00000000..72403c32 --- /dev/null +++ b/tunnels/shared/wireguard/wireguard.c @@ -0,0 +1,1098 @@ +/* + Most of the code is taken and renamed, from the awesome projects wireguard-lwip and lwip + + Author of lwip: Adam Dunkels https://github.com/smartalock/wireguard-lwip + Author of wireguard-lwip: Daniel Hope https://github.com/lwip-tcpip/lwip + + their license files are placed next to this file +*/ + +#include "wireguard.h" + +#include +#include +#include +#include + +#include "crypto.h" + +// For HMAC calculation +#define WIREGUARD_BLAKE2S_BLOCK_SIZE (64) + +// 5.4 Messages +// Constants +static const uint8_t CONSTRUCTION[37] = "Noise_IKpsk2_25519_ChaChaPoly_BLAKE2s"; // The UTF-8 string literal "Noise_IKpsk2_25519_ChaChaPoly_BLAKE2s", 37 bytes of output +static const uint8_t IDENTIFIER[34] = "WireGuard v1 zx2c4 Jason@zx2c4.com"; // The UTF-8 string literal "WireGuard v1 zx2c4 Jason@zx2c4.com", 34 bytes of output +static const uint8_t LABEL_MAC1[8] = "mac1----"; // Label-Mac1 The UTF-8 string literal "mac1----", 8 bytes of output. +static const uint8_t LABEL_COOKIE[8] = "cookie--"; // Label-Cookie The UTF-8 string literal "cookie--", 8 bytes of output + +static const char *base64_lookup = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; + +static const uint8_t zero_key[WIREGUARD_PUBLIC_KEY_LEN] = { 0 }; + +// Calculated in wireguard_init +static uint8_t construction_hash[WIREGUARD_HASH_LEN]; +static uint8_t identifier_hash[WIREGUARD_HASH_LEN]; + + +void wireguard_init() { + wireguard_blake2s_ctx ctx; + // Pre-calculate chaining key hash + wireguard_blake2s_init(&ctx, WIREGUARD_HASH_LEN, NULL, 0); + wireguard_blake2s_update(&ctx, CONSTRUCTION, sizeof(CONSTRUCTION)); + wireguard_blake2s_final(&ctx, construction_hash); + // Pre-calculate initial handshake hash - uses construction_hash calculated above + wireguard_blake2s_init(&ctx, WIREGUARD_HASH_LEN, NULL, 0); + wireguard_blake2s_update(&ctx, construction_hash, sizeof(construction_hash)); + wireguard_blake2s_update(&ctx, IDENTIFIER, sizeof(IDENTIFIER)); + wireguard_blake2s_final(&ctx, identifier_hash); +} + +struct wireguard_peer *peer_alloc(struct wireguard_device *device) { + struct wireguard_peer *result = NULL; + struct wireguard_peer *tmp; + int x; + for (x=0; x < WIREGUARD_MAX_PEERS; x++) { + tmp = &device->peers[x]; + if (!tmp->valid) { + result = tmp; + break; + } + } + return result; +} + +struct wireguard_peer *peer_lookup_by_pubkey(struct wireguard_device *device, uint8_t *public_key) { + struct wireguard_peer *result = NULL; + struct wireguard_peer *tmp; + int x; + for (x=0; x < WIREGUARD_MAX_PEERS; x++) { + tmp = &device->peers[x]; + if (tmp->valid) { + if (memcmp(tmp->public_key, public_key, WIREGUARD_PUBLIC_KEY_LEN) == 0) { + result = tmp; + break; + } + } + } + return result; +} + +uint8_t wireguard_peer_index(struct wireguard_device *device, struct wireguard_peer *peer) { + uint8_t result = 0xFF; + uint8_t x; + for (x=0; x < WIREGUARD_MAX_PEERS; x++) { + if (peer == &device->peers[x]) { + result = x; + break; + } + } + return result; +} + +struct wireguard_peer *peer_lookup_by_peer_index(struct wireguard_device *device, uint8_t peer_index) { + struct wireguard_peer *result = NULL; + if (peer_index < WIREGUARD_MAX_PEERS) { + if (device->peers[peer_index].valid) { + result = &device->peers[peer_index]; + } + } + return result; +} + +struct wireguard_peer *peer_lookup_by_receiver(struct wireguard_device *device, uint32_t receiver) { + struct wireguard_peer *result = NULL; + struct wireguard_peer *tmp; + int x; + for (x=0; x < WIREGUARD_MAX_PEERS; x++) { + tmp = &device->peers[x]; + if (tmp->valid) { + if ((tmp->curr_keypair.valid && (tmp->curr_keypair.local_index == receiver)) || + (tmp->next_keypair.valid && (tmp->next_keypair.local_index == receiver)) || + (tmp->prev_keypair.valid && (tmp->prev_keypair.local_index == receiver)) + ) { + result = tmp; + break; + } + } + } + return result; +} + +struct wireguard_peer *peer_lookup_by_handshake(struct wireguard_device *device, uint32_t receiver) { + struct wireguard_peer *result = NULL; + struct wireguard_peer *tmp; + int x; + for (x=0; x < WIREGUARD_MAX_PEERS; x++) { + tmp = &device->peers[x]; + if (tmp->valid) { + if (tmp->handshake.valid && tmp->handshake.initiator && (tmp->handshake.local_index == receiver)) { + result = tmp; + break; + } + } + } + return result; +} + +bool wireguard_expired(uint32_t created_millis, uint32_t valid_seconds) { + uint32_t diff = wireguard_sys_now() - created_millis; + return (diff >= (valid_seconds * 1000)); +} + + +static void generate_cookie_secret(struct wireguard_device *device) { + wireguard_random_bytes(device->cookie_secret, WIREGUARD_HASH_LEN); + device->cookie_secret_millis = wireguard_sys_now(); +} + +static void generate_peer_cookie(struct wireguard_device *device, uint8_t *cookie, uint8_t *source_addr_port, size_t source_length) { + wireguard_blake2s_ctx ctx; + + if (wireguard_expired(device->cookie_secret_millis, COOKIE_SECRET_MAX_AGE)) { + // Generate new random bytes + generate_cookie_secret(device); + } + + // Mac(key, input) Keyed-Blake2s(key, input, 16), the keyed MAC variant of the BLAKE2s hash function, returning 16 bytes of output + wireguard_blake2s_init(&ctx, WIREGUARD_COOKIE_LEN, device->cookie_secret, WIREGUARD_HASH_LEN); + // 5.4.7 Under Load: Cookie Reply Message + // Mix in the IP address and port - have the IP layer pass this in as byte array to avoid using Lwip specific APIs in this module + if ((source_addr_port) && (source_length > 0)) { + wireguard_blake2s_update(&ctx, source_addr_port, source_length); + } + wireguard_blake2s_final(&ctx, cookie); +} + +static void wireguard_mac(uint8_t *dst, const void *message, size_t len, const uint8_t *key, size_t keylen) { + wireguard_blake2s(dst, WIREGUARD_COOKIE_LEN, key, keylen, message, len); +} + +static void wireguard_mac_key(uint8_t *key, const uint8_t *public_key, const uint8_t *label, size_t label_len) { + blake2s_ctx ctx; + blake2s_init(&ctx, WIREGUARD_SESSION_KEY_LEN, NULL, 0); + blake2s_update(&ctx, label, label_len); + blake2s_update(&ctx, public_key, WIREGUARD_PUBLIC_KEY_LEN); + blake2s_final(&ctx, key); +} + +static void wireguard_mix_hash(uint8_t *hash, const uint8_t *src, size_t src_len) { + wireguard_blake2s_ctx ctx; + wireguard_blake2s_init(&ctx, WIREGUARD_HASH_LEN, NULL, 0); + wireguard_blake2s_update(&ctx, hash, WIREGUARD_HASH_LEN); + wireguard_blake2s_update(&ctx, src, src_len); + wireguard_blake2s_final(&ctx, hash); +} + +static void wireguard_hmac(uint8_t *digest, const uint8_t *key, size_t key_len, const uint8_t *text, size_t text_len) { + // Adapted from appendix example in RFC2104 to use BLAKE2S instead of MD5 - https://tools.ietf.org/html/rfc2104 + wireguard_blake2s_ctx ctx; + uint8_t k_ipad[WIREGUARD_BLAKE2S_BLOCK_SIZE]; // inner padding - key XORd with ipad + uint8_t k_opad[WIREGUARD_BLAKE2S_BLOCK_SIZE]; // outer padding - key XORd with opad + + uint8_t tk[WIREGUARD_HASH_LEN]; + int i; + // if key is longer than BLAKE2S_BLOCK_SIZE bytes reset it to key=BLAKE2S(key) + if (key_len > WIREGUARD_BLAKE2S_BLOCK_SIZE) { + wireguard_blake2s_ctx tctx; + wireguard_blake2s_init(&tctx, WIREGUARD_HASH_LEN, NULL, 0); + wireguard_blake2s_update(&tctx, key, key_len); + wireguard_blake2s_final(&tctx, tk); + key = tk; + key_len = WIREGUARD_HASH_LEN; + } + + // the HMAC transform looks like: + // HASH(K XOR opad, HASH(K XOR ipad, text)) + // where K is an n byte key + // ipad is the byte 0x36 repeated BLAKE2S_BLOCK_SIZE times + // opad is the byte 0x5c repeated BLAKE2S_BLOCK_SIZE times + // and text is the data being protected + memset(k_ipad, 0, sizeof(k_ipad)); + memset(k_opad, 0, sizeof(k_opad)); + memcpy(k_ipad, key, key_len); + memcpy(k_opad, key, key_len); + + // XOR key with ipad and opad values + for (i=0; i < WIREGUARD_BLAKE2S_BLOCK_SIZE; i++) { + k_ipad[i] ^= 0x36; + k_opad[i] ^= 0x5c; + } + // perform inner HASH + wireguard_blake2s_init(&ctx, WIREGUARD_HASH_LEN, NULL, 0); // init context for 1st pass + wireguard_blake2s_update(&ctx, k_ipad, WIREGUARD_BLAKE2S_BLOCK_SIZE); // start with inner pad + wireguard_blake2s_update(&ctx, text, text_len); // then text of datagram + wireguard_blake2s_final(&ctx, digest); // finish up 1st pass + + // perform outer HASH + wireguard_blake2s_init(&ctx, WIREGUARD_HASH_LEN, NULL, 0); // init context for 2nd pass + wireguard_blake2s_update(&ctx, k_opad, WIREGUARD_BLAKE2S_BLOCK_SIZE); // start with outer pad + wireguard_blake2s_update(&ctx, digest, WIREGUARD_HASH_LEN); // then results of 1st hash + wireguard_blake2s_final(&ctx, digest); // finish up 2nd pass +} + +static void wireguard_kdf1(uint8_t *tau1, const uint8_t *chaining_key, const uint8_t *data, size_t data_len) { + uint8_t tau0[WIREGUARD_HASH_LEN]; + uint8_t output[WIREGUARD_HASH_LEN + 1]; + + // tau0 = Hmac(key, input) + wireguard_hmac(tau0, chaining_key, WIREGUARD_HASH_LEN, data, data_len); + // tau1 := Hmac(tau0, 0x1) + output[0] = 1; + wireguard_hmac(output, tau0, WIREGUARD_HASH_LEN, output, 1); + memcpy(tau1, output, WIREGUARD_HASH_LEN); + + // Wipe intermediates + crypto_zero(tau0, sizeof(tau0)); + crypto_zero(output, sizeof(output)); +} + +static void wireguard_kdf2(uint8_t *tau1, uint8_t *tau2, const uint8_t *chaining_key, const uint8_t *data, size_t data_len) { + uint8_t tau0[WIREGUARD_HASH_LEN]; + uint8_t output[WIREGUARD_HASH_LEN + 1]; + + // tau0 = Hmac(key, input) + wireguard_hmac(tau0, chaining_key, WIREGUARD_HASH_LEN, data, data_len); + // tau1 := Hmac(tau0, 0x1) + output[0] = 1; + wireguard_hmac(output, tau0, WIREGUARD_HASH_LEN, output, 1); + memcpy(tau1, output, WIREGUARD_HASH_LEN); + + // tau2 := Hmac(tau0,tau1 || 0x2) + output[WIREGUARD_HASH_LEN] = 2; + wireguard_hmac(output, tau0, WIREGUARD_HASH_LEN, output, WIREGUARD_HASH_LEN + 1); + memcpy(tau2, output, WIREGUARD_HASH_LEN); + + // Wipe intermediates + crypto_zero(tau0, sizeof(tau0)); + crypto_zero(output, sizeof(output)); +} + +static void wireguard_kdf3(uint8_t *tau1, uint8_t *tau2, uint8_t *tau3, const uint8_t *chaining_key, const uint8_t *data, size_t data_len) { + uint8_t tau0[WIREGUARD_HASH_LEN]; + uint8_t output[WIREGUARD_HASH_LEN + 1]; + + // tau0 = Hmac(key, input) + wireguard_hmac(tau0, chaining_key, WIREGUARD_HASH_LEN, data, data_len); + // tau1 := Hmac(tau0, 0x1) + output[0] = 1; + wireguard_hmac(output, tau0, WIREGUARD_HASH_LEN, output, 1); + memcpy(tau1, output, WIREGUARD_HASH_LEN); + + // tau2 := Hmac(tau0,tau1 || 0x2) + output[WIREGUARD_HASH_LEN] = 2; + wireguard_hmac(output, tau0, WIREGUARD_HASH_LEN, output, WIREGUARD_HASH_LEN + 1); + memcpy(tau2, output, WIREGUARD_HASH_LEN); + + // tau3 := Hmac(tau0,tau1,tau2 || 0x3) + output[WIREGUARD_HASH_LEN] = 3; + wireguard_hmac(output, tau0, WIREGUARD_HASH_LEN, output, WIREGUARD_HASH_LEN + 1); + memcpy(tau3, output, WIREGUARD_HASH_LEN); + + // Wipe intermediates + crypto_zero(tau0, sizeof(tau0)); + crypto_zero(output, sizeof(output)); +} + +bool wireguard_check_replay(struct wireguard_keypair *keypair, uint64_t seq) { + // Implementation of packet replay window - as per RFC2401 + // Adapted from code in Appendix C at https://tools.ietf.org/html/rfc2401 + uint32_t diff; + bool result = false; + size_t ReplayWindowSize = sizeof(keypair->replay_bitmap) * CHAR_BIT; // 32 bits + + // WireGuard data packet counter starts from 0 but algorithm expects packet numbers to start from 1 + seq++; + + if (seq != 0) { + if (seq > keypair->replay_counter) { + // new larger sequence number + diff = seq - keypair->replay_counter; + if (diff < ReplayWindowSize) { + // In window + keypair->replay_bitmap <<= diff; + // set bit for this packet + keypair->replay_bitmap |= 1; + } else { + // This packet has a "way larger" + keypair->replay_bitmap = 1; + } + keypair->replay_counter = seq; + // larger is good + result = true; + } else { + diff = keypair->replay_counter - seq; + if (diff < ReplayWindowSize) { + if (keypair->replay_bitmap & ((uint32_t)1 << diff)) { + // already seen + } else { + // mark as seen + keypair->replay_bitmap |= ((uint32_t)1 << diff); + // out of order but good + result = true; + } + } else { + // too old or wrapped + } + } + } else { + // first == 0 or wrapped + } + return result; +} + +struct wireguard_keypair *get_peer_keypair_for_idx(struct wireguard_peer *peer, uint32_t idx) { + if (peer->curr_keypair.valid && peer->curr_keypair.local_index == idx) { + return &peer->curr_keypair; + } else if (peer->next_keypair.valid && peer->next_keypair.local_index == idx) { + return &peer->next_keypair; + } else if (peer->prev_keypair.valid && peer->prev_keypair.local_index == idx) { + return &peer->prev_keypair; + } + return NULL; +} + +static uint32_t wireguard_generate_unique_index(struct wireguard_device *device) { + // We need a random 32-bit number but make sure it's not already been used in the context of this device + uint32_t result; + uint8_t buf[4]; + int x; + struct wireguard_peer *peer; + bool existing; + do { + do { + wireguard_random_bytes(buf, 4); + result = U8TO32_LITTLE(buf); + } while ((result == 0) || (result == 0xFFFFFFFF)); // Don't allow 0 or 0xFFFFFFFF as valid values + + existing = false; + for (x=0; x < WIREGUARD_MAX_PEERS; x++) { + peer = &device->peers[x]; + existing = (result == peer->curr_keypair.local_index) || + (result == peer->prev_keypair.local_index) || + (result == peer->next_keypair.local_index) || + (result == peer->handshake.local_index); + + } + } while (existing); + + return result; +} + +static void wireguard_clamp_private_key(uint8_t *key) { + key[0] &= 248; + key[31] = (key[31] & 127) | 64; +} + +static void wireguard_generate_private_key(uint8_t *key) { + wireguard_random_bytes(key, WIREGUARD_PRIVATE_KEY_LEN); + wireguard_clamp_private_key(key); +} + +static bool wireguard_generate_public_key(uint8_t *public_key, const uint8_t *private_key) { + static const uint8_t basepoint[WIREGUARD_PUBLIC_KEY_LEN] = { 9 }; + bool result = false; + if (memcmp(private_key, zero_key, WIREGUARD_PUBLIC_KEY_LEN) != 0) { + result = (wireguard_x25519(public_key, private_key, basepoint) == 0); + } + return result; +} + +bool wireguard_check_mac1(struct wireguard_device *device, const uint8_t *data, size_t len, const uint8_t *mac1) { + bool result = false; + uint8_t calculated[WIREGUARD_COOKIE_LEN]; + wireguard_mac(calculated, data, len, device->label_mac1_key, WIREGUARD_SESSION_KEY_LEN); + if (crypto_equal(calculated, mac1, WIREGUARD_COOKIE_LEN)) { + result = true; + } + return result; +} + +bool wireguard_check_mac2(struct wireguard_device *device, const uint8_t *data, size_t len, uint8_t *source_addr_port, size_t source_length, const uint8_t *mac2) { + bool result = false; + uint8_t cookie[WIREGUARD_COOKIE_LEN]; + uint8_t calculated[WIREGUARD_COOKIE_LEN]; + + generate_peer_cookie(device, cookie, source_addr_port, source_length); + + wireguard_mac(calculated, data, len, cookie, WIREGUARD_COOKIE_LEN); + if (crypto_equal(calculated, mac2, WIREGUARD_COOKIE_LEN)) { + result = true; + } + return result; +} + +void keypair_destroy(struct wireguard_keypair *keypair) { + crypto_zero(keypair, sizeof(struct wireguard_keypair)); + keypair->valid = false; +} + +void keypair_update(struct wireguard_peer *peer, struct wireguard_keypair *received_keypair) { + bool key_is_next = (received_keypair == &peer->next_keypair); + if (key_is_next) { + peer->prev_keypair = peer->curr_keypair; + peer->curr_keypair = peer->next_keypair; + keypair_destroy(&peer->next_keypair); + } +} + +static void add_new_keypair(struct wireguard_peer *peer, struct wireguard_keypair new_keypair) { + if (new_keypair.initiator) { + if (peer->next_keypair.valid) { + peer->prev_keypair = peer->next_keypair; + keypair_destroy(&peer->next_keypair); + } else { + peer->prev_keypair = peer->curr_keypair; + } + peer->curr_keypair = new_keypair; + } else { + peer->next_keypair = new_keypair; + keypair_destroy(&peer->prev_keypair); + } +} + +void wireguard_start_session(struct wireguard_peer *peer, bool initiator) { + struct wireguard_handshake *handshake = &peer->handshake; + struct wireguard_keypair new_keypair; + + crypto_zero(&new_keypair, sizeof(struct wireguard_keypair)); + new_keypair.initiator = initiator; + new_keypair.local_index = handshake->local_index; + new_keypair.remote_index = handshake->remote_index; + + new_keypair.keypair_millis = wireguard_sys_now(); + new_keypair.sending_valid = true; + new_keypair.receiving_valid = true; + + // 5.4.5 Transport Data Key Derivation + // (Tsendi = Trecvr, Trecvi = Tsendr) := Kdf2(Ci = Cr,E) + if (new_keypair.initiator) { + wireguard_kdf2(new_keypair.sending_key, new_keypair.receiving_key, handshake->chaining_key, NULL, 0); + } else { + wireguard_kdf2(new_keypair.receiving_key, new_keypair.sending_key, handshake->chaining_key, NULL, 0); + } + + new_keypair.replay_bitmap = 0; + new_keypair.replay_counter = 0; + + new_keypair.last_tx = 0; + new_keypair.last_rx = 0; // No packets received yet + + new_keypair.valid = true; + + // Eprivi = Epubi = Eprivr = Epubr = Ci = Cr := E + crypto_zero(handshake->ephemeral_private, WIREGUARD_PUBLIC_KEY_LEN); + crypto_zero(handshake->remote_ephemeral, WIREGUARD_PUBLIC_KEY_LEN); + crypto_zero(handshake->hash, WIREGUARD_HASH_LEN); + crypto_zero(handshake->chaining_key, WIREGUARD_HASH_LEN); + handshake->remote_index = 0; + handshake->local_index = 0; + handshake->valid = false; + + add_new_keypair(peer, new_keypair); +} + +uint8_t wireguard_get_message_type(const uint8_t *data, size_t len) { + uint8_t result = MESSAGE_INVALID; + if (len >= 4) { + if ((data[1] == 0) && (data[2] == 0) && (data[3] == 0)) { + switch (data[0]) { + case MESSAGE_HANDSHAKE_INITIATION: + if (len == sizeof(struct message_handshake_initiation)) { + result = MESSAGE_HANDSHAKE_INITIATION; + } + break; + case MESSAGE_HANDSHAKE_RESPONSE: + if (len == sizeof(struct message_handshake_response)) { + result = MESSAGE_HANDSHAKE_RESPONSE; + } + break; + case MESSAGE_COOKIE_REPLY: + if (len == sizeof(struct message_cookie_reply)) { + result = MESSAGE_COOKIE_REPLY; + } + break; + case MESSAGE_TRANSPORT_DATA: + if (len >= sizeof(struct message_transport_data) + WIREGUARD_AUTHTAG_LEN) { + result = MESSAGE_TRANSPORT_DATA; + } + break; + default: + break; + } + } + } + return result; +} + +struct wireguard_peer *wireguard_process_initiation_message(struct wireguard_device *device, struct message_handshake_initiation *msg) { + struct wireguard_peer *ret_peer = NULL; + struct wireguard_peer *peer = NULL; + struct wireguard_handshake *handshake; + uint8_t key[WIREGUARD_SESSION_KEY_LEN]; + uint8_t chaining_key[WIREGUARD_HASH_LEN]; + uint8_t hash[WIREGUARD_HASH_LEN]; + uint8_t s[WIREGUARD_PUBLIC_KEY_LEN]; + uint8_t e[WIREGUARD_PUBLIC_KEY_LEN]; + uint8_t t[WIREGUARD_TAI64N_LEN]; + uint8_t dh_calculation[WIREGUARD_PUBLIC_KEY_LEN]; + uint32_t now; + bool rate_limit; + bool replay; + + // We are the responder, other end is the initiator + + // Ci := Hash(Construction) (precalculated hash) + memcpy(chaining_key, construction_hash, WIREGUARD_HASH_LEN); + + // Hi := Hash(Ci || Identifier + memcpy(hash, identifier_hash, WIREGUARD_HASH_LEN); + + // Hi := Hash(Hi || Spubr) + wireguard_mix_hash(hash, device->public_key, WIREGUARD_PUBLIC_KEY_LEN); + + // Ci := Kdf1(Ci, Epubi) + wireguard_kdf1(chaining_key, chaining_key, msg->ephemeral, WIREGUARD_PUBLIC_KEY_LEN); + + // msg.ephemeral := Epubi + memcpy(e, msg->ephemeral, WIREGUARD_PUBLIC_KEY_LEN); + + // Hi := Hash(Hi || msg.ephemeral) + wireguard_mix_hash(hash, msg->ephemeral, WIREGUARD_PUBLIC_KEY_LEN); + + // Calculate DH(Eprivi,Spubr) + wireguard_x25519(dh_calculation, device->private_key, e); + if (!crypto_equal(dh_calculation, zero_key, WIREGUARD_PUBLIC_KEY_LEN)) { + + // (Ci,k) := Kdf2(Ci,DH(Eprivi,Spubr)) + wireguard_kdf2(chaining_key, key, chaining_key, dh_calculation, WIREGUARD_PUBLIC_KEY_LEN); + + // msg.static := AEAD(k, 0, Spubi, Hi) + if (wireguard_aead_decrypt(s, msg->enc_static, sizeof(msg->enc_static), hash, WIREGUARD_HASH_LEN, 0, key)) { + // Hi := Hash(Hi || msg.static) + wireguard_mix_hash(hash, msg->enc_static, sizeof(msg->enc_static)); + + peer = peer_lookup_by_pubkey(device, s); + if (peer) { + handshake = &peer->handshake; + + // (Ci,k) := Kdf2(Ci,DH(Sprivi,Spubr)) + wireguard_kdf2(chaining_key, key, chaining_key, peer->public_key_dh, WIREGUARD_PUBLIC_KEY_LEN); + + // msg.timestamp := AEAD(k, 0, Timestamp(), Hi) + if (wireguard_aead_decrypt(t, msg->enc_timestamp, sizeof(msg->enc_timestamp), hash, WIREGUARD_HASH_LEN, 0, key)) { + // Hi := Hash(Hi || msg.timestamp) + wireguard_mix_hash(hash, msg->enc_timestamp, sizeof(msg->enc_timestamp)); + + now = wireguard_sys_now(); + + // Check that timestamp is increasing and we haven't had too many initiations (should only get one per peer every 5 seconds max?) + replay = (memcmp(t, peer->greatest_timestamp, WIREGUARD_TAI64N_LEN) <= 0); // tai64n is big endian so we can use memcmp to compare + rate_limit = (peer->last_initiation_rx - now) < (1000 / MAX_INITIATIONS_PER_SECOND); + + if (!replay && !rate_limit) { + // Success! Copy everything to peer + peer->last_initiation_rx = now; + if (memcmp(t, peer->greatest_timestamp, WIREGUARD_TAI64N_LEN) > 0) { + memcpy(peer->greatest_timestamp, t, WIREGUARD_TAI64N_LEN); + // TODO: Need to notify if the higher layers want to persist latest timestamp/nonce somewhere + } + memcpy(handshake->remote_ephemeral, e, WIREGUARD_PUBLIC_KEY_LEN); + memcpy(handshake->hash, hash, WIREGUARD_HASH_LEN); + memcpy(handshake->chaining_key, chaining_key, WIREGUARD_HASH_LEN); + handshake->remote_index = msg->sender; + handshake->valid = true; + handshake->initiator = false; + ret_peer = peer; + + } else { + // Ignore + } + } else { + // Failed to decrypt + } + } else { + // peer not found + } + } else { + // Failed to decrypt + } + } else { + // Bad X25519 + } + + crypto_zero(key, sizeof(key)); + crypto_zero(hash, sizeof(hash)); + crypto_zero(chaining_key, sizeof(chaining_key)); + crypto_zero(dh_calculation, sizeof(dh_calculation)); + + return ret_peer; +} + +bool wireguard_process_handshake_response(struct wireguard_device *device, struct wireguard_peer *peer, struct message_handshake_response *src) { + struct wireguard_handshake *handshake = &peer->handshake; + + bool result = false; + uint8_t key[WIREGUARD_SESSION_KEY_LEN]; + uint8_t hash[WIREGUARD_HASH_LEN]; + uint8_t chaining_key[WIREGUARD_HASH_LEN]; + uint8_t e[WIREGUARD_PUBLIC_KEY_LEN]; + uint8_t ephemeral_private[WIREGUARD_PUBLIC_KEY_LEN]; + uint8_t static_private[WIREGUARD_PUBLIC_KEY_LEN]; + uint8_t preshared_key[WIREGUARD_SESSION_KEY_LEN]; + uint8_t dh_calculation[WIREGUARD_PUBLIC_KEY_LEN]; + uint8_t tau[WIREGUARD_PUBLIC_KEY_LEN]; + + if (handshake->valid && handshake->initiator) { + + memcpy(hash, handshake->hash, WIREGUARD_HASH_LEN); + memcpy(chaining_key, handshake->chaining_key, WIREGUARD_HASH_LEN); + memcpy(ephemeral_private, handshake->ephemeral_private, WIREGUARD_PUBLIC_KEY_LEN); + memcpy(preshared_key, peer->preshared_key, WIREGUARD_SESSION_KEY_LEN); + + // (Eprivr, Epubr) := DH-Generate() + // Not required + + // Cr := Kdf1(Cr,Epubr) + wireguard_kdf1(chaining_key, chaining_key, src->ephemeral, WIREGUARD_PUBLIC_KEY_LEN); + + // msg.ephemeral := Epubr + memcpy(e, src->ephemeral, WIREGUARD_PUBLIC_KEY_LEN); + + // Hr := Hash(Hr || msg.ephemeral) + wireguard_mix_hash(hash, src->ephemeral, WIREGUARD_PUBLIC_KEY_LEN); + + // Cr := Kdf1(Cr, DH(Eprivr, Epubi)) + // Calculate DH(Eprivr, Epubi) + wireguard_x25519(dh_calculation, ephemeral_private, e); + if (!crypto_equal(dh_calculation, zero_key, WIREGUARD_PUBLIC_KEY_LEN)) { + wireguard_kdf1(chaining_key, chaining_key, dh_calculation, WIREGUARD_PUBLIC_KEY_LEN); + + // Cr := Kdf1(Cr, DH(Eprivr, Spubi)) + // CalculateDH(Eprivr, Spubi) + wireguard_x25519(dh_calculation, device->private_key, e); + if (!crypto_equal(dh_calculation, zero_key, WIREGUARD_PUBLIC_KEY_LEN)) { + wireguard_kdf1(chaining_key, chaining_key, dh_calculation, WIREGUARD_PUBLIC_KEY_LEN); + + // (Cr, t, k) := Kdf3(Cr, Q) + wireguard_kdf3(chaining_key, tau, key, chaining_key, peer->preshared_key, WIREGUARD_SESSION_KEY_LEN); + + // Hr := Hash(Hr | t) + wireguard_mix_hash(hash, tau, WIREGUARD_HASH_LEN); + + // msg.empty := AEAD(k, 0, E, Hr) + if (wireguard_aead_decrypt(NULL, src->enc_empty, sizeof(src->enc_empty), hash, WIREGUARD_HASH_LEN, 0, key)) { + // Hr := Hash(Hr | msg.empty) + // Not required as discarded + + //Copy details to handshake + memcpy(handshake->remote_ephemeral, e, WIREGUARD_HASH_LEN); + memcpy(handshake->hash, hash, WIREGUARD_HASH_LEN); + memcpy(handshake->chaining_key, chaining_key, WIREGUARD_HASH_LEN); + handshake->remote_index = src->sender; + + result = true; + } else { + // Decrypt failed + } + + } else { + // X25519 fail + } + + } else { + // X25519 fail + } + + } + crypto_zero(key, sizeof(key)); + crypto_zero(hash, sizeof(hash)); + crypto_zero(chaining_key, sizeof(chaining_key)); + crypto_zero(ephemeral_private, sizeof(ephemeral_private)); + crypto_zero(static_private, sizeof(static_private)); + crypto_zero(preshared_key, sizeof(preshared_key)); + crypto_zero(tau, sizeof(tau)); + + return result; +} + +bool wireguard_process_cookie_message(struct wireguard_device *device, struct wireguard_peer *peer, struct message_cookie_reply *src) { + uint8_t cookie[WIREGUARD_COOKIE_LEN]; + bool result = false; + + if (peer->handshake_mac1_valid) { + + result = wireguard_xaead_decrypt(cookie, src->enc_cookie, sizeof(src->enc_cookie), peer->handshake_mac1, WIREGUARD_COOKIE_LEN, src->nonce, peer->label_cookie_key); + + if (result) { + // 5.4.7 Under Load: Cookie Reply Message + // Upon receiving this message, if it is valid, the only thing the recipient of this message should do is store the cookie along with the time at which it was received + memcpy(peer->cookie, cookie, WIREGUARD_COOKIE_LEN); + peer->cookie_millis = wireguard_sys_now(); + peer->handshake_mac1_valid = false; + } + } else { + // We didn't send any initiation packet so we shouldn't be getting a cookie reply! + } + return result; +} + +bool wireguard_create_handshake_initiation(struct wireguard_device *device, struct wireguard_peer *peer, struct message_handshake_initiation *dst) { + uint8_t timestamp[WIREGUARD_TAI64N_LEN]; + uint8_t key[WIREGUARD_SESSION_KEY_LEN]; + uint8_t dh_calculation[WIREGUARD_PUBLIC_KEY_LEN]; + bool result = false; + + struct wireguard_handshake *handshake = &peer->handshake; + + memset(dst, 0, sizeof(struct message_handshake_initiation)); + + // Ci := Hash(Construction) (precalculated hash) + memcpy(handshake->chaining_key, construction_hash, WIREGUARD_HASH_LEN); + + // Hi := Hash(Ci || Identifier) + memcpy(handshake->hash, identifier_hash, WIREGUARD_HASH_LEN); + + // Hi := Hash(Hi || Spubr) + wireguard_mix_hash(handshake->hash, peer->public_key, WIREGUARD_PUBLIC_KEY_LEN); + + // (Eprivi, Epubi) := DH-Generate() + wireguard_generate_private_key(handshake->ephemeral_private); + if (wireguard_generate_public_key(dst->ephemeral, handshake->ephemeral_private)) { + + // Ci := Kdf1(Ci, Epubi) + wireguard_kdf1(handshake->chaining_key, handshake->chaining_key, dst->ephemeral, WIREGUARD_PUBLIC_KEY_LEN); + + // msg.ephemeral := Epubi + // Done above - public keys is calculated into dst->ephemeral + + // Hi := Hash(Hi || msg.ephemeral) + wireguard_mix_hash(handshake->hash, dst->ephemeral, WIREGUARD_PUBLIC_KEY_LEN); + + // Calculate DH(Eprivi,Spubr) + wireguard_x25519(dh_calculation, handshake->ephemeral_private, peer->public_key); + if (!crypto_equal(dh_calculation, zero_key, WIREGUARD_PUBLIC_KEY_LEN)) { + + // (Ci,k) := Kdf2(Ci,DH(Eprivi,Spubr)) + wireguard_kdf2(handshake->chaining_key, key, handshake->chaining_key, dh_calculation, WIREGUARD_PUBLIC_KEY_LEN); + + // msg.static := AEAD(k,0,Spubi, Hi) + wireguard_aead_encrypt(dst->enc_static, device->public_key, WIREGUARD_PUBLIC_KEY_LEN, handshake->hash, WIREGUARD_HASH_LEN, 0, key); + + // Hi := Hash(Hi || msg.static) + wireguard_mix_hash(handshake->hash, dst->enc_static, sizeof(dst->enc_static)); + + // (Ci,k) := Kdf2(Ci,DH(Sprivi,Spubr)) + // note DH(Sprivi,Spubr) is precomputed per peer + wireguard_kdf2(handshake->chaining_key, key, handshake->chaining_key, peer->public_key_dh, WIREGUARD_PUBLIC_KEY_LEN); + + // msg.timestamp := AEAD(k, 0, Timestamp(), Hi) + wireguard_tai64n_now(timestamp); + wireguard_aead_encrypt(dst->enc_timestamp, timestamp, WIREGUARD_TAI64N_LEN, handshake->hash, WIREGUARD_HASH_LEN, 0, key); + + // Hi := Hash(Hi || msg.timestamp) + wireguard_mix_hash(handshake->hash, dst->enc_timestamp, sizeof(dst->enc_timestamp)); + + dst->type = MESSAGE_HANDSHAKE_INITIATION; + dst->sender = wireguard_generate_unique_index(device); + + handshake->valid = true; + handshake->initiator = true; + handshake->local_index = dst->sender; + + result = true; + } + } + + if (result) { + // 5.4.4 Cookie MACs + // msg.mac1 := Mac(Hash(Label-Mac1 || Spubm' ), msgA) + // The value Hash(Label-Mac1 || Spubm' ) above can be pre-computed + wireguard_mac(dst->mac1, dst, (sizeof(struct message_handshake_initiation)-(2*WIREGUARD_COOKIE_LEN)), peer->label_mac1_key, WIREGUARD_SESSION_KEY_LEN); + + // if Lm = E or Lm ≥ 120: + if ((peer->cookie_millis == 0) || wireguard_expired(peer->cookie_millis, COOKIE_SECRET_MAX_AGE)) { + // msg.mac2 := 0 + crypto_zero(dst->mac2, WIREGUARD_COOKIE_LEN); + } else { + // msg.mac2 := Mac(Lm, msgB) + wireguard_mac(dst->mac2, dst, (sizeof(struct message_handshake_initiation)-(WIREGUARD_COOKIE_LEN)), peer->cookie, WIREGUARD_COOKIE_LEN); + + } + } + + crypto_zero(key, sizeof(key)); + crypto_zero(dh_calculation, sizeof(dh_calculation)); + return result; +} + +bool wireguard_create_handshake_response(struct wireguard_device *device, struct wireguard_peer *peer, struct message_handshake_response *dst) { + struct wireguard_handshake *handshake = &peer->handshake; + uint8_t key[WIREGUARD_SESSION_KEY_LEN]; + uint8_t dh_calculation[WIREGUARD_PUBLIC_KEY_LEN]; + uint8_t tau[WIREGUARD_HASH_LEN]; + bool result = false; + + memset(dst, 0, sizeof(struct message_handshake_response)); + + if (handshake->valid && !handshake->initiator) { + + // (Eprivr, Epubr) := DH-Generate() + wireguard_generate_private_key(handshake->ephemeral_private); + if (wireguard_generate_public_key(dst->ephemeral, handshake->ephemeral_private)) { + + // Cr := Kdf1(Cr,Epubr) + wireguard_kdf1(handshake->chaining_key, handshake->chaining_key, dst->ephemeral, WIREGUARD_PUBLIC_KEY_LEN); + + // msg.ephemeral := Epubr + // Copied above when generated + + // Hr := Hash(Hr || msg.ephemeral) + wireguard_mix_hash(handshake->hash, dst->ephemeral, WIREGUARD_PUBLIC_KEY_LEN); + + // Cr := Kdf1(Cr, DH(Eprivr, Epubi)) + // Calculate DH(Eprivi,Spubr) + wireguard_x25519(dh_calculation, handshake->ephemeral_private, handshake->remote_ephemeral); + if (!crypto_equal(dh_calculation, zero_key, WIREGUARD_PUBLIC_KEY_LEN)) { + wireguard_kdf1(handshake->chaining_key, handshake->chaining_key, dh_calculation, WIREGUARD_PUBLIC_KEY_LEN); + + // Cr := Kdf1(Cr, DH(Eprivr, Spubi)) + // Calculate DH(Eprivi,Spubr) + wireguard_x25519(dh_calculation, handshake->ephemeral_private, peer->public_key); + if (!crypto_equal(dh_calculation, zero_key, WIREGUARD_PUBLIC_KEY_LEN)) { + wireguard_kdf1(handshake->chaining_key, handshake->chaining_key, dh_calculation, WIREGUARD_PUBLIC_KEY_LEN); + + // (Cr, t, k) := Kdf3(Cr, Q) + wireguard_kdf3(handshake->chaining_key, tau, key, handshake->chaining_key, peer->preshared_key, WIREGUARD_SESSION_KEY_LEN); + + // Hr := Hash(Hr | t) + wireguard_mix_hash(handshake->hash, tau, WIREGUARD_HASH_LEN); + + // msg.empty := AEAD(k, 0, E, Hr) + wireguard_aead_encrypt(dst->enc_empty, NULL, 0, handshake->hash, WIREGUARD_HASH_LEN, 0, key); + + // Hr := Hash(Hr | msg.empty) + wireguard_mix_hash(handshake->hash, dst->enc_empty, sizeof(dst->enc_empty)); + + dst->type = MESSAGE_HANDSHAKE_RESPONSE; + dst->receiver = handshake->remote_index; + dst->sender = wireguard_generate_unique_index(device); + // Update handshake object too + handshake->local_index = dst->sender; + + result = true; + } else { + // Bad x25519 + } + } else { + // Bad x25519 + } + + } else { + // Failed to generate DH + } + } + + if (result) { + // 5.4.4 Cookie MACs + // msg.mac1 := Mac(Hash(Label-Mac1 || Spubm' ), msgA) + // The value Hash(Label-Mac1 || Spubm' ) above can be pre-computed + wireguard_mac(dst->mac1, dst, (sizeof(struct message_handshake_response)-(2*WIREGUARD_COOKIE_LEN)), peer->label_mac1_key, WIREGUARD_SESSION_KEY_LEN); + + // if Lm = E or Lm ≥ 120: + if ((peer->cookie_millis == 0) || wireguard_expired(peer->cookie_millis, COOKIE_SECRET_MAX_AGE)) { + // msg.mac2 := 0 + crypto_zero(dst->mac2, WIREGUARD_COOKIE_LEN); + } else { + // msg.mac2 := Mac(Lm, msgB) + wireguard_mac(dst->mac2, dst, (sizeof(struct message_handshake_response)-(WIREGUARD_COOKIE_LEN)), peer->cookie, WIREGUARD_COOKIE_LEN); + } + } + + crypto_zero(key, sizeof(key)); + crypto_zero(dh_calculation, sizeof(dh_calculation)); + crypto_zero(tau, sizeof(tau)); + return result; +} + +void wireguard_create_cookie_reply(struct wireguard_device *device, struct message_cookie_reply *dst, const uint8_t *mac1, uint32_t index, uint8_t *source_addr_port, size_t source_length) { + uint8_t cookie[WIREGUARD_COOKIE_LEN]; + crypto_zero(dst, sizeof(struct message_cookie_reply)); + dst->type = MESSAGE_COOKIE_REPLY; + dst->receiver = index; + wireguard_random_bytes(dst->nonce, COOKIE_NONCE_LEN); + generate_peer_cookie(device, cookie, source_addr_port, source_length); + wireguard_xaead_encrypt(dst->enc_cookie, cookie, WIREGUARD_COOKIE_LEN, mac1, WIREGUARD_COOKIE_LEN, dst->nonce, device->label_cookie_key); +} + +bool wireguard_peer_init(struct wireguard_device *device, struct wireguard_peer *peer, const uint8_t *public_key, const uint8_t *preshared_key) { + // Clear out structure + memset(peer, 0, sizeof(struct wireguard_peer)); + + if (device->valid) { + // Copy across the public key into our peer structure + memcpy(peer->public_key, public_key, WIREGUARD_PUBLIC_KEY_LEN); + if (preshared_key) { + memcpy(peer->preshared_key, preshared_key, WIREGUARD_SESSION_KEY_LEN); + } else { + crypto_zero(peer->preshared_key, WIREGUARD_SESSION_KEY_LEN); + } + + if (wireguard_x25519(peer->public_key_dh, device->private_key, peer->public_key) == 0) { + // Zero out handshake + memset(&peer->handshake, 0, sizeof(struct wireguard_handshake)); + peer->handshake.valid = false; + + // Zero out any cookie info - we haven't received one yet + peer->cookie_millis = 0; + memset(&peer->cookie, 0, WIREGUARD_COOKIE_LEN); + + // Precompute keys to deal with mac1/2 calculation + wireguard_mac_key(peer->label_mac1_key, peer->public_key, LABEL_MAC1, sizeof(LABEL_MAC1)); + wireguard_mac_key(peer->label_cookie_key, peer->public_key, LABEL_COOKIE, sizeof(LABEL_COOKIE)); + + peer->valid = true; + } else { + crypto_zero(peer->public_key_dh, WIREGUARD_PUBLIC_KEY_LEN); + } + } + return peer->valid; +} + +bool wireguard_device_init(struct wireguard_device *device, const uint8_t *private_key) { + // Set the private key and calculate public key from it + memcpy(device->private_key, private_key, WIREGUARD_PRIVATE_KEY_LEN); + // Ensure private key is correctly "clamped" + wireguard_clamp_private_key(device->private_key); + device->valid = wireguard_generate_public_key(device->public_key, private_key); + if (device->valid) { + generate_cookie_secret(device); + // 5.4.4 Cookie MACs - The value Hash(Label-Mac1 || Spubm' ) above can be pre-computed. + wireguard_mac_key(device->label_mac1_key, device->public_key, LABEL_MAC1, sizeof(LABEL_MAC1)); + // 5.4.7 Under Load: Cookie Reply Message - The value Hash(Label-Cookie || Spubm) above can be pre-computed. + wireguard_mac_key(device->label_cookie_key, device->public_key, LABEL_COOKIE, sizeof(LABEL_COOKIE)); + + } else { + crypto_zero(device->private_key, WIREGUARD_PRIVATE_KEY_LEN); + } + return device->valid; +} + +void wireguard_encrypt_packet(uint8_t *dst, const uint8_t *src, size_t src_len, struct wireguard_keypair *keypair) { + wireguard_aead_encrypt(dst, src, src_len, NULL, 0, keypair->sending_counter, keypair->sending_key); + keypair->sending_counter++; +} + +bool wireguard_decrypt_packet(uint8_t *dst, const uint8_t *src, size_t src_len, uint64_t counter, struct wireguard_keypair *keypair) { + return wireguard_aead_decrypt(dst, src, src_len, NULL, 0, counter, keypair->receiving_key); +} + +bool wireguard_base64_decode(const char *str, uint8_t *out, size_t *outlen) { + uint32_t accum = 0; // We accumulate upto four blocks of 6 bits into this to form 3 bytes output + uint8_t char_count = 0; // How many characters have we processed in this block + int byte_count = 3; // How many bytes are we expecting in current 4 char block + int len = 0; // result length in bytes + bool result = true; + uint8_t bits; + char c; + char *ptr; + int x; + size_t inlen; + + if (!str) { + return false; + } + + inlen = strlen(str); + + for (x = 0; x < inlen; x++) { + c = str[x]; + if (c == '=') { + // This is '=' padding at end of string - decrease the number of bytes to write + bits = 0; + byte_count--; + if (byte_count < 0) { + // Too much padding! + result = false; + break; + } + } else { + if (byte_count != 3) { + // Padding only allowed at end - this is a valid byte and we have already seen padding + result = false; + break; + } + ptr = strchr(base64_lookup, c); + if (ptr) { + bits = (uint8_t)((ptr - base64_lookup) & 0x3F); + } else { + // invalid character in input string + result = false; + break; + } + } + + accum = (accum << 6) | bits; + char_count++; + + if (char_count == 4) { + if (len + byte_count > *outlen) { + // Output buffer overflow + result = false; + break; + } + out[len++] = (uint8_t)((accum >> 16) & 0xFF); + if (byte_count > 1) { + out[len++] = (uint8_t)((accum >> 8) & 0xFF); + } + if (byte_count > 2) { + out[len++] = (uint8_t)(accum & 0xFF); + } + char_count = 0; + accum = 0; + } + } + if (char_count != 0) { + // We require padding to multiple of 3 input length - bytes are missing from output! + result = false; + } + *outlen = len; + return result; +} + +bool wireguard_base64_encode(const uint8_t *in, size_t inlen, char *out, size_t *outlen) { + bool result = false; + int read_offset = 0; + int write_offset = 0; + uint8_t byte1, byte2, byte3; + uint32_t tmp; + char c; + size_t len = 4 * ((inlen + 2) / 3); + int padding = (3 - (inlen % 3)); + if (padding > 2) padding = 0; + if (*outlen > len) { + + while (read_offset < inlen) { + // Read three bytes + byte1 = (read_offset < inlen) ? in[read_offset++] : 0; + byte2 = (read_offset < inlen) ? in[read_offset++] : 0; + byte3 = (read_offset < inlen) ? in[read_offset++] : 0; + // Turn into 24 bit intermediate + tmp = (byte1 << 16) | (byte2 << 8) | (byte3); + // Write out 4 characters each representing 6 bits of input + out[write_offset++] = base64_lookup[(tmp >> 18) & 0x3F]; + out[write_offset++] = base64_lookup[(tmp >> 12) & 0x3F]; + c = (write_offset < len - padding) ? base64_lookup[(tmp >> 6) & 0x3F] : '='; + out[write_offset++] = c; + c = (write_offset < len - padding) ? base64_lookup[(tmp) & 0x3F] : '='; + out[write_offset++] = c; + } + out[len] = '\0'; + *outlen = len; + result = true; + } else { + // Not enough data to put in base64 and null terminate + } + return result; +} + diff --git a/tunnels/shared/wireguard/wireguard.h b/tunnels/shared/wireguard/wireguard.h new file mode 100644 index 00000000..2cca7e2e --- /dev/null +++ b/tunnels/shared/wireguard/wireguard.h @@ -0,0 +1,55 @@ +#pragma once + +/* + Most of the code is taken and renamed, from the awesome projects wireguard-lwip and lwip + + Author of lwip: Adam Dunkels https://github.com/smartalock/wireguard-lwip + Author of wireguard-lwip: Daniel Hope https://github.com/lwip-tcpip/lwip + + their license files are placed next to this file +*/ + +#include "defs.h" + +// Initialise the WireGuard system - need to call this before anything else +void wireguard_init(); +bool wireguard_device_init(struct wireguard_device *device, const uint8_t *private_key); +bool wireguard_peer_init(struct wireguard_device *device, struct wireguard_peer *peer, const uint8_t *public_key, const uint8_t *preshared_key); + +struct wireguard_peer *peer_alloc(struct wireguard_device *device); +uint8_t wireguard_peer_index(struct wireguard_device *device, struct wireguard_peer *peer); +struct wireguard_peer *peer_lookup_by_pubkey(struct wireguard_device *device, uint8_t *public_key); +struct wireguard_peer *peer_lookup_by_peer_index(struct wireguard_device *device, uint8_t peer_index); +struct wireguard_peer *peer_lookup_by_receiver(struct wireguard_device *device, uint32_t receiver); +struct wireguard_peer *peer_lookup_by_handshake(struct wireguard_device *device, uint32_t receiver); + +void wireguard_start_session(struct wireguard_peer *peer, bool initiator); + +void keypair_update(struct wireguard_peer *peer, struct wireguard_keypair *received_keypair); +void keypair_destroy(struct wireguard_keypair *keypair); + +struct wireguard_keypair *get_peer_keypair_for_idx(struct wireguard_peer *peer, uint32_t idx); +bool wireguard_check_replay(struct wireguard_keypair *keypair, uint64_t seq); + +uint8_t wireguard_get_message_type(const uint8_t *data, size_t len); + +struct wireguard_peer *wireguard_process_initiation_message(struct wireguard_device *device, struct message_handshake_initiation *msg); +bool wireguard_process_handshake_response(struct wireguard_device *device, struct wireguard_peer *peer, struct message_handshake_response *src); +bool wireguard_process_cookie_message(struct wireguard_device *device, struct wireguard_peer *peer, struct message_cookie_reply *src); + +bool wireguard_create_handshake_initiation(struct wireguard_device *device, struct wireguard_peer *peer, struct message_handshake_initiation *dst); +bool wireguard_create_handshake_response(struct wireguard_device *device, struct wireguard_peer *peer, struct message_handshake_response *dst); +void wireguard_create_cookie_reply(struct wireguard_device *device, struct message_cookie_reply *dst, const uint8_t *mac1, uint32_t index, uint8_t *source_addr_port, size_t source_length); + + +bool wireguard_check_mac1(struct wireguard_device *device, const uint8_t *data, size_t len, const uint8_t *mac1); +bool wireguard_check_mac2(struct wireguard_device *device, const uint8_t *data, size_t len, uint8_t *source_addr_port, size_t source_length, const uint8_t *mac2); + +bool wireguard_expired(uint32_t created_millis, uint32_t valid_seconds); + +void wireguard_encrypt_packet(uint8_t *dst, const uint8_t *src, size_t src_len, struct wireguard_keypair *keypair); +bool wireguard_decrypt_packet(uint8_t *dst, const uint8_t *src, size_t src_len, uint64_t counter, struct wireguard_keypair *keypair); + +bool wireguard_base64_decode(const char *str, uint8_t *out, size_t *outlen); +bool wireguard_base64_encode(const uint8_t *in, size_t inlen, char *out, size_t *outlen); + diff --git a/tunnels/shared/wireguard/wireguardif.h b/tunnels/shared/wireguard/wireguardif.h new file mode 100644 index 00000000..36503b85 --- /dev/null +++ b/tunnels/shared/wireguard/wireguardif.h @@ -0,0 +1,108 @@ +#pragma once +/* + Most of the code is taken and renamed, from the awesome projects wireguard-lwip and lwip + + Author of lwip: Adam Dunkels https://github.com/smartalock/wireguard-lwip + Author of wireguard-lwip: Daniel Hope https://github.com/lwip-tcpip/lwip + + their license files are placed next to this file +*/ + +#include "lwip_types.h" + +// Default MTU for WireGuard is 1420 bytes + +enum +{ + kWgIFMtu = 1420, + kWgDefaultPort = 51820, + kWgDefaultKeepAlive = 0xFFFF, + kWgIFInvalidIndex = 0xFF +}; + +typedef struct wireguardif_init_data_s +{ + // Required: the private key of this WireGuard network interface + const char *private_key; + // Required: What UDP port to listen on + uint16_t listen_port; + // Optional: restrict send/receive of encapsulated WireGuard traffic to this network interface only (NULL to use + // routing table) + struct netif *bind_netif; + +} wireguardif_init_data_t; + +typedef struct wireguardif_peer_s +{ + const char *public_key; + // Optional pre-shared key (32 bytes) - make sure this is NULL if not to be used + const uint8_t *preshared_key; + // tai64n of largest timestamp we have seen during handshake to avoid replays + uint8_t greatest_timestamp[12]; + + // Allowed ip/netmask (can add additional later but at least one is required) + ip_addr_t allowed_ip; + ip_addr_t allowed_mask; + + // End-point details (may be blank) + ip_addr_t endpoint_ip; + uint16_t endport_port; + uint16_t keep_alive; + +} wireguardif_peer_t; + +/* static struct netif wg_netif_struct = {0}; + * struct wireguard_interface wg; + * wg.private_key = "abcdefxxx..xxxxx="; + * wg.listen_port = 51820; + * wg.bind_netif = NULL; // Pass netif to listen on, NULL for all interfaces + * + * netif = netif_add(&netif_struct, &ipaddr, &netmask, &gateway, &wg, &wireguardif_init, &ip_input); + * + * netif_set_up(wg_net); + * + * struct wireguardif_peer peer; + * wireguardif_peer_init(&peer); + * peer.public_key = "apoehc...4322abcdfejg=; + * peer.preshared_key = NULL; + * peer.allowed_ip = allowed_ip; + * peer.allowed_mask = allowed_mask; + * + * // If you want to enable output connection + * peer.endpoint_ip = peer_ip; + * peer.endport_port = 12345; + * + * uint8_t wireguard_peer_index; + * wireguardif_add_peer(netif, &peer, &wireguard_peer_index); + * + * if ((wireguard_peer_index != WIREGUARDIF_INVALID_INDEX) && !ip_addr_isany(&peer.endpoint_ip)) { + * // Start outbound connection to peer + * wireguardif_connect(wg_net, wireguard_peer_index); + * } + * + */ + +// Initialise a new WireGuard network interface (netif) +err_t wireguardIFInit(struct netif *netif); + +// Helper to initialise the peer struct with defaults +void wireguardIFPeerInit(struct wireguardif_peer_s *peer); + +// Add a new peer to the specified interface - see wireguard.h for maximum number of peers allowed +// On success the peer_index can be used to reference this peer in future function calls +err_t wireguardIFAddPeer(struct netif *netif, struct wireguardif_peer_s *peer, uint8_t *peer_index); + +// Remove the given peer from the network interface +err_t wireguardIFRemovePeer(struct netif *netif, uint8_t peer_index); + +// Update the "connect" IP of the given peer +err_t wireguardIFUpdateEndpoint(struct netif *netif, uint8_t peer_index, const ip_addr_t *ip, uint16_t port); + +// Try and connect to the given peer +err_t wireguardIFConnect(struct netif *netif, uint8_t peer_index); + +// Stop trying to connect to the given peer +err_t wireguardIFDisconnect(struct netif *netif, uint8_t peer_index); + +// Is the given peer "up"? A peer is up if it has a valid session key it can communicate with +err_t wireguardIFPeerIsUp(struct netif *netif, uint8_t peer_index, ip_addr_t *current_ip, uint16_t *current_port);