| /* SPDX-License-Identifier: GPL-2.0 */ | 
 | /* | 
 |  * Copyright (C) 2015-2019 Jason A. Donenfeld <Jason@zx2c4.com>. All Rights Reserved. | 
 |  */ | 
 | #ifndef _WG_NOISE_H | 
 | #define _WG_NOISE_H | 
 |  | 
 | #include "messages.h" | 
 | #include "peerlookup.h" | 
 |  | 
 | #include <linux/types.h> | 
 | #include <linux/spinlock.h> | 
 | #include <linux/atomic.h> | 
 | #include <linux/rwsem.h> | 
 | #include <linux/mutex.h> | 
 | #include <linux/kref.h> | 
 |  | 
 | struct noise_replay_counter { | 
 | 	u64 counter; | 
 | 	spinlock_t lock; | 
 | 	unsigned long backtrack[COUNTER_BITS_TOTAL / BITS_PER_LONG]; | 
 | }; | 
 |  | 
 | struct noise_symmetric_key { | 
 | 	u8 key[NOISE_SYMMETRIC_KEY_LEN]; | 
 | 	u64 birthdate; | 
 | 	bool is_valid; | 
 | }; | 
 |  | 
 | struct noise_keypair { | 
 | 	struct index_hashtable_entry entry; | 
 | 	struct noise_symmetric_key sending; | 
 | 	atomic64_t sending_counter; | 
 | 	struct noise_symmetric_key receiving; | 
 | 	struct noise_replay_counter receiving_counter; | 
 | 	__le32 remote_index; | 
 | 	bool i_am_the_initiator; | 
 | 	struct kref refcount; | 
 | 	struct rcu_head rcu; | 
 | 	u64 internal_id; | 
 | }; | 
 |  | 
 | struct noise_keypairs { | 
 | 	struct noise_keypair __rcu *current_keypair; | 
 | 	struct noise_keypair __rcu *previous_keypair; | 
 | 	struct noise_keypair __rcu *next_keypair; | 
 | 	spinlock_t keypair_update_lock; | 
 | }; | 
 |  | 
 | struct noise_static_identity { | 
 | 	u8 static_public[NOISE_PUBLIC_KEY_LEN]; | 
 | 	u8 static_private[NOISE_PUBLIC_KEY_LEN]; | 
 | 	struct rw_semaphore lock; | 
 | 	bool has_identity; | 
 | }; | 
 |  | 
 | enum noise_handshake_state { | 
 | 	HANDSHAKE_ZEROED, | 
 | 	HANDSHAKE_CREATED_INITIATION, | 
 | 	HANDSHAKE_CONSUMED_INITIATION, | 
 | 	HANDSHAKE_CREATED_RESPONSE, | 
 | 	HANDSHAKE_CONSUMED_RESPONSE | 
 | }; | 
 |  | 
 | struct noise_handshake { | 
 | 	struct index_hashtable_entry entry; | 
 |  | 
 | 	enum noise_handshake_state state; | 
 | 	u64 last_initiation_consumption; | 
 |  | 
 | 	struct noise_static_identity *static_identity; | 
 |  | 
 | 	u8 ephemeral_private[NOISE_PUBLIC_KEY_LEN]; | 
 | 	u8 remote_static[NOISE_PUBLIC_KEY_LEN]; | 
 | 	u8 remote_ephemeral[NOISE_PUBLIC_KEY_LEN]; | 
 | 	u8 precomputed_static_static[NOISE_PUBLIC_KEY_LEN]; | 
 |  | 
 | 	u8 preshared_key[NOISE_SYMMETRIC_KEY_LEN]; | 
 |  | 
 | 	u8 hash[NOISE_HASH_LEN]; | 
 | 	u8 chaining_key[NOISE_HASH_LEN]; | 
 |  | 
 | 	u8 latest_timestamp[NOISE_TIMESTAMP_LEN]; | 
 | 	__le32 remote_index; | 
 |  | 
 | 	/* Protects all members except the immutable (after noise_handshake_ | 
 | 	 * init): remote_static, precomputed_static_static, static_identity. | 
 | 	 */ | 
 | 	struct rw_semaphore lock; | 
 | }; | 
 |  | 
 | struct wg_device; | 
 |  | 
 | void wg_noise_init(void); | 
 | void wg_noise_handshake_init(struct noise_handshake *handshake, | 
 | 			     struct noise_static_identity *static_identity, | 
 | 			     const u8 peer_public_key[NOISE_PUBLIC_KEY_LEN], | 
 | 			     const u8 peer_preshared_key[NOISE_SYMMETRIC_KEY_LEN], | 
 | 			     struct wg_peer *peer); | 
 | void wg_noise_handshake_clear(struct noise_handshake *handshake); | 
 | static inline void wg_noise_reset_last_sent_handshake(atomic64_t *handshake_ns) | 
 | { | 
 | 	atomic64_set(handshake_ns, ktime_get_coarse_boottime_ns() - | 
 | 				       (u64)(REKEY_TIMEOUT + 1) * NSEC_PER_SEC); | 
 | } | 
 |  | 
 | void wg_noise_keypair_put(struct noise_keypair *keypair, bool unreference_now); | 
 | struct noise_keypair *wg_noise_keypair_get(struct noise_keypair *keypair); | 
 | void wg_noise_keypairs_clear(struct noise_keypairs *keypairs); | 
 | bool wg_noise_received_with_keypair(struct noise_keypairs *keypairs, | 
 | 				    struct noise_keypair *received_keypair); | 
 | void wg_noise_expire_current_peer_keypairs(struct wg_peer *peer); | 
 |  | 
 | void wg_noise_set_static_identity_private_key( | 
 | 	struct noise_static_identity *static_identity, | 
 | 	const u8 private_key[NOISE_PUBLIC_KEY_LEN]); | 
 | void wg_noise_precompute_static_static(struct wg_peer *peer); | 
 |  | 
 | bool | 
 | wg_noise_handshake_create_initiation(struct message_handshake_initiation *dst, | 
 | 				     struct noise_handshake *handshake); | 
 | struct wg_peer * | 
 | wg_noise_handshake_consume_initiation(struct message_handshake_initiation *src, | 
 | 				      struct wg_device *wg); | 
 |  | 
 | bool wg_noise_handshake_create_response(struct message_handshake_response *dst, | 
 | 					struct noise_handshake *handshake); | 
 | struct wg_peer * | 
 | wg_noise_handshake_consume_response(struct message_handshake_response *src, | 
 | 				    struct wg_device *wg); | 
 |  | 
 | bool wg_noise_handshake_begin_session(struct noise_handshake *handshake, | 
 | 				      struct noise_keypairs *keypairs); | 
 |  | 
 | #endif /* _WG_NOISE_H */ |