GNUnet 0.25.2-11-g84e94e98c
 
Loading...
Searching...
No Matches
gnunet-service-core_kx.h File Reference

code for managing the key exchange (SET_KEY, PING, PONG) with other peers More...

Include dependency graph for gnunet-service-core_kx.h:
This graph shows which files directly or indirectly include this file:

Go to the source code of this file.

Data Structures

struct  InitiatorHelloPayload
 
struct  InitiatorHello
 TODO. More...
 
struct  ConfirmationAck
 The ACK. More...
 
struct  ResponderHello
 ResponderHello. More...
 
struct  ResponderHelloPayload
 
struct  InitiatorDone
 InitiatorDone. More...
 
struct  EncryptedMessage
 EncryptedMessage. More...
 
struct  Heartbeat
 KeyUpdate. More...
 

Enumerations

enum  HeartbeatFlags { GSC_HEARTBEAT_KEY_UPDATE_REQUESTED = 1 }
 Heartbeat flags. More...
 

Functions

void GSC_KX_encrypt_and_transmit (struct GSC_KeyExchangeInfo *kx, const void *payload, size_t payload_size)
 Encrypt and transmit a message with the given payload.
 
int GSC_KX_init (void)
 Initialize KX subsystem.
 
void GSC_KX_done (void)
 Shutdown KX subsystem.
 
int GSC_NEIGHBOURS_check_excess_bandwidth (const struct GSC_KeyExchangeInfo *target)
 Check if the given neighbour has excess bandwidth available.
 
unsigned int GSC_NEIGHBOURS_get_queue_length (const struct GSC_KeyExchangeInfo *target)
 Check how many messages are queued for the given neighbour.
 
void GSC_KX_handle_client_monitor_peers (struct GNUNET_MQ_Handle *mq)
 Handle GNUNET_MESSAGE_TYPE_CORE_MONITOR_PEERS request.
 

Detailed Description

code for managing the key exchange (SET_KEY, PING, PONG) with other peers

Author
Christian Grothoff

Definition in file gnunet-service-core_kx.h.

Enumeration Type Documentation

◆ HeartbeatFlags

Heartbeat flags.

Enumerator
GSC_HEARTBEAT_KEY_UPDATE_REQUESTED 

A key update is requested.

Definition at line 225 of file gnunet-service-core_kx.h.

226{
231
232};
@ GSC_HEARTBEAT_KEY_UPDATE_REQUESTED
A key update is requested.

Function Documentation

◆ GSC_KX_encrypt_and_transmit()

void GSC_KX_encrypt_and_transmit ( struct GSC_KeyExchangeInfo kx,
const void *  payload,
size_t  payload_size 
)

Encrypt and transmit a message with the given payload.

Parameters
kxkey exchange context
payloadpayload of the message
payload_sizenumber of bytes in 'payload'

Encrypt and transmit a message with the given payload.

Parameters
kxkey exchange info
payloadthe payload
payload_sizesize of the payload

Definition at line 2746 of file gnunet-service-core_kx.c.

2749{
2750 struct GNUNET_MQ_Envelope *env;
2751 struct EncryptedMessage *encrypted_msg;
2752 unsigned char enc_key[AEAD_KEY_BYTES];
2753 unsigned char enc_nonce[AEAD_NONCE_BYTES];
2754 unsigned char seq_enc_k[crypto_stream_chacha20_ietf_KEYBYTES];
2755 uint64_t sqn;
2756 uint64_t epoch;
2757 int8_t ret;
2758
2759 encrypted_msg = NULL;
2760
2761 check_rekey (kx);
2762 sqn = kx->current_sqn;
2763 epoch = kx->current_epoch;
2764 /* We are the sender and as we are going to send,
2765 * we are using the initiator key material */
2767 sqn,
2768 enc_key,
2769 enc_nonce);
2770 kx->current_sqn++;
2771 derive_sn (&kx->current_ats,
2772 seq_enc_k,
2773 sizeof seq_enc_k);
2774 env = GNUNET_MQ_msg_extra (encrypted_msg,
2775 payload_size,
2777 // only encrypt the payload for now
2778 // TODO encrypt other fields as well
2779 ret = crypto_aead_xchacha20poly1305_ietf_encrypt_detached (
2780 (unsigned char*) &encrypted_msg[1], // c - resulting ciphertext
2781 (unsigned char*) &encrypted_msg->tag, // mac - resulting mac/tag
2782 NULL, // maclen
2783 (unsigned char*) payload, // m - plain message
2784 payload_size, // mlen
2785 NULL, // ad - additional data TODO also cover the unencrypted part (epoch)
2786 0, // adlen
2787 NULL, // nsec - unused
2788 enc_nonce, // npub nonce
2789 enc_key // k - key
2790 );
2791 if (0 != ret)
2792 {
2794 "Something went wrong encrypting message\n");
2795 GNUNET_assert (0);
2796 }
2797 {
2798 /* compute the sequence number */
2799 unsigned char *seq_enc_nonce;
2800 uint64_t seq_nbo;
2801 uint32_t seq_enc_ctr;
2802
2803 seq_nbo = GNUNET_htonll (sqn);
2804 seq_enc_ctr = *((uint32_t*) encrypted_msg->tag);
2805 seq_enc_nonce = &encrypted_msg->tag[sizeof (uint32_t)];
2806 crypto_stream_chacha20_ietf_xor_ic (
2807 (unsigned char*) &encrypted_msg->sequence_number,
2808 (unsigned char*) &seq_nbo,
2809 sizeof seq_nbo,
2810 seq_enc_nonce,
2811 ntohl (seq_enc_ctr),
2812 seq_enc_k);
2813#if DEBUG_KX
2814 GNUNET_print_bytes (seq_enc_k,
2815 sizeof seq_enc_k,
2816 8,
2817 GNUNET_NO);
2818 GNUNET_print_bytes ((char*) &seq_enc_ctr,
2819 sizeof seq_enc_ctr,
2820 8,
2821 GNUNET_NO);
2822#endif
2824 "Sending encrypted message with E(SQN=%" PRIu64 ")=%" PRIu64
2825 "\n",
2826 sqn,
2827 encrypted_msg->sequence_number);
2828 }
2829 encrypted_msg->epoch = GNUNET_htonll (epoch);
2830
2831 // TODO actually copy payload
2832 GNUNET_MQ_send (kx->mq, env);
2833}
struct GNUNET_MQ_Envelope * env
Definition 005.c:1
static int ret
Final status code.
Definition gnunet-arm.c:93
#define AEAD_NONCE_BYTES
libsodium has very long symbol names
static void check_rekey(struct GSC_KeyExchangeInfo *kx)
static void derive_per_message_secrets(const struct GNUNET_ShortHashCode *ts, uint64_t seq, unsigned char key[crypto_aead_xchacha20poly1305_ietf_KEYBYTES], unsigned char nonce[crypto_aead_xchacha20poly1305_ietf_NPUBBYTES])
key = HKDF-Expand [I,R][A,H]TS, "key", 32) nonce = HKDF-Expand ([I,R][A,H]TS, "iv",...
#define AEAD_KEY_BYTES
libsodium has very long symbol names
static void derive_sn(const struct GNUNET_ShortHashCode *secret, unsigned char *sn, size_t sn_len)
static unsigned long long payload
How much data are we currently storing in the database?
#define GNUNET_log(kind,...)
uint64_t GNUNET_htonll(uint64_t n)
Convert unsigned 64-bit integer to network byte order.
@ GNUNET_NO
void GNUNET_print_bytes(const void *buf, size_t buf_len, int fold, int in_be)
Print a byte string in hexadecimal ascii notation.
#define GNUNET_assert(cond)
Use this for fatal errors that cannot be handled.
@ GNUNET_ERROR_TYPE_ERROR
@ GNUNET_ERROR_TYPE_DEBUG
void GNUNET_MQ_send(struct GNUNET_MQ_Handle *mq, struct GNUNET_MQ_Envelope *ev)
Send a message with the given message queue.
Definition mq.c:305
#define GNUNET_MQ_msg_extra(mvar, esize, type)
Allocate an envelope, with extra space allocated after the space needed by the message struct.
#define GNUNET_MESSAGE_TYPE_CORE_ENCRYPTED_MESSAGE_CAKE
Encrypted message.
uint64_t current_sqn
Our current sequence number.
struct GNUNET_MQ_Handle * mq
Message queue for sending messages to peer.
uint64_t current_epoch
Our currently used epoch for sending.
struct GNUNET_ShortHashCode current_ats
*ATS - our current application traffic secret by epoch

References AEAD_KEY_BYTES, AEAD_NONCE_BYTES, check_rekey(), GSC_KeyExchangeInfo::current_ats, GSC_KeyExchangeInfo::current_epoch, GSC_KeyExchangeInfo::current_sqn, derive_per_message_secrets(), derive_sn(), env, EncryptedMessage::epoch, GNUNET_assert, GNUNET_ERROR_TYPE_DEBUG, GNUNET_ERROR_TYPE_ERROR, GNUNET_htonll(), GNUNET_log, GNUNET_MESSAGE_TYPE_CORE_ENCRYPTED_MESSAGE_CAKE, GNUNET_MQ_msg_extra, GNUNET_MQ_send(), GNUNET_NO, GNUNET_print_bytes(), GSC_KeyExchangeInfo::mq, payload, ret, EncryptedMessage::sequence_number, and EncryptedMessage::tag.

Referenced by handle_heartbeat(), handle_initiator_done(), send_heartbeat(), and try_transmission().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSC_KX_init()

int GSC_KX_init ( void  )

Initialize KX subsystem.

Returns
GNUNET_OK on success, GNUNET_SYSERR on failure

Definition at line 2870 of file gnunet-service-core_kx.c.

2871{
2873 GNUNET_MQ_hd_var_size (initiator_hello,
2875 struct InitiatorHello,
2876 NULL),
2877 GNUNET_MQ_hd_var_size (initiator_done,
2879 struct InitiatorDone,
2880 NULL),
2881 GNUNET_MQ_hd_var_size (responder_hello,
2883 struct ResponderHello,
2884 NULL),
2885 GNUNET_MQ_hd_var_size (encrypted_message, // TODO rename?
2887 struct EncryptedMessage,
2888 NULL),
2890 };
2891
2895 NULL); // TODO potentially wait
2896 // until we have a peer_id?
2897 // pay attention to whether
2898 // we have one anyways
2899 if (NULL == pils)
2900 {
2901 GSC_KX_done ();
2902 return GNUNET_SYSERR;
2903 }
2904
2906 transport =
2909 handlers,
2910 NULL, // cls - this connection-independant
2911 // cls seems not to be needed.
2912 // the connection-specific cls
2913 // will be set as a return value
2914 // of
2915 // handle_transport_notify_connect
2918 if (NULL == transport)
2919 {
2920 GSC_KX_done ();
2921 return GNUNET_SYSERR;
2922 }
2924 "Connected to TRANSPORT\n");
2925 return GNUNET_OK;
2926}
struct GNUNET_MQ_MessageHandlers handlers[]
Definition 003.c:1
const struct GNUNET_CONFIGURATION_Handle * GSC_cfg
Our configuration.
struct GNUNET_PeerIdentity GSC_my_identity
Our identity.
static void * handle_transport_notify_connect(void *cls, const struct GNUNET_PeerIdentity *peer_id, struct GNUNET_MQ_Handle *mq)
Function called by transport to notify us that a peer connected to us (on the network level).
static void peer_id_change_cb(void *cls, const struct GNUNET_HELLO_Parser *parser, const struct GNUNET_HashCode *hash)
Callback for PILS to be called once the peer id changes.
static void handle_transport_notify_disconnect(void *cls, const struct GNUNET_PeerIdentity *peer, void *handler_cls)
Function called by transport telling us that a peer disconnected.
static struct GNUNET_NotificationContext * nc
Notification context for broadcasting to monitors.
void GSC_KX_done()
Shutdown KX subsystem.
static enum GNUNET_GenericReturnValue init_phase
Indicates whether we are still in the initialisation phase (waiting for our peer id).
static struct GNUNET_TRANSPORT_CoreHandle * transport
Transport service.
static struct GNUNET_PILS_Handle * pils
Pils service.
struct GNUNET_PILS_Handle * GNUNET_PILS_connect(const struct GNUNET_CONFIGURATION_Handle *cfg, GNUNET_PILS_PidChangeCallback pid_change_cb, void *cls)
Connect to the PILS service.
Definition pils_api.c:367
struct GNUNET_TRANSPORT_CoreHandle * GNUNET_TRANSPORT_core_connect(const struct GNUNET_CONFIGURATION_Handle *cfg, const struct GNUNET_PeerIdentity *self, const struct GNUNET_MQ_MessageHandler *handlers, void *cls, GNUNET_TRANSPORT_NotifyConnect nc, GNUNET_TRANSPORT_NotifyDisconnect nd)
Connect to the transport service.
@ GNUNET_OK
@ GNUNET_YES
@ GNUNET_SYSERR
#define GNUNET_MQ_handler_end()
End-marker for the handlers array.
struct GNUNET_NotificationContext * GNUNET_notification_context_create(unsigned int queue_length)
Create a new notification context.
Definition nc.c:122
#define GNUNET_MQ_hd_var_size(name, code, str, ctx)
#define GNUNET_MESSAGE_TYPE_CORE_INITIATOR_DONE
Third and final message of the handshake, second of the initiator.
#define GNUNET_MESSAGE_TYPE_CORE_RESPONDER_HELLO
Reply to the first message from the initiator - first message sent by the responder.
#define GNUNET_MESSAGE_TYPE_CORE_INITIATOR_HELLO
for more detail on the following messages see https://lsd.gnunet.org/lsd0012/draft-schanzen-cake....
Message handler for a specific message type.

References GNUNET_ERROR_TYPE_DEBUG, GNUNET_log, GNUNET_MESSAGE_TYPE_CORE_ENCRYPTED_MESSAGE_CAKE, GNUNET_MESSAGE_TYPE_CORE_INITIATOR_DONE, GNUNET_MESSAGE_TYPE_CORE_INITIATOR_HELLO, GNUNET_MESSAGE_TYPE_CORE_RESPONDER_HELLO, GNUNET_MQ_handler_end, GNUNET_MQ_hd_var_size, GNUNET_notification_context_create(), GNUNET_OK, GNUNET_PILS_connect(), GNUNET_SYSERR, GNUNET_TRANSPORT_core_connect(), GNUNET_YES, GSC_cfg, GSC_KX_done(), GSC_my_identity, handle_transport_notify_connect(), handle_transport_notify_disconnect(), handlers, init_phase, nc, peer_id_change_cb(), pils, and transport.

Referenced by run().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSC_KX_done()

void GSC_KX_done ( void  )

Shutdown KX subsystem.

Definition at line 2933 of file gnunet-service-core_kx.c.

2934{
2935 struct PilsRequest *pr;
2936 while (NULL != (pr = pils_requests_head))
2937 {
2940 pr);
2941 if (NULL != pr->op)
2942 GNUNET_PILS_cancel (pr->op);
2943 GNUNET_free (pr);
2944 }
2945 if (NULL != pils)
2946 {
2948 pils = NULL;
2949 }
2950 if (NULL != transport)
2951 {
2953 transport = NULL;
2954 }
2955 if (NULL != rekey_task)
2956 {
2958 rekey_task = NULL;
2959 }
2960 if (NULL != nc)
2961 {
2963 nc = NULL;
2964 }
2965}
static struct PilsRequest * pils_requests_head
PILS Operation DLL.
static struct PilsRequest * pils_requests_tail
PILS Operation DLL.
static struct GNUNET_SCHEDULER_Task * rekey_task
Task scheduled for periodic re-generation (and thus rekeying) of our ephemeral key.
void GNUNET_PILS_disconnect(struct GNUNET_PILS_Handle *handle)
Disconnect from the PILS service.
Definition pils_api.c:390
void GNUNET_PILS_cancel(struct GNUNET_PILS_Operation *op)
Cancel request.
Definition pils_api.c:495
void GNUNET_TRANSPORT_core_disconnect(struct GNUNET_TRANSPORT_CoreHandle *handle)
Disconnect from the transport service.
#define GNUNET_CONTAINER_DLL_remove(head, tail, element)
Remove an element from a DLL.
#define GNUNET_free(ptr)
Wrapper around free.
void GNUNET_notification_context_destroy(struct GNUNET_NotificationContext *nc)
Destroy the context, force disconnect for all subscribers.
Definition nc.c:138
void * GNUNET_SCHEDULER_cancel(struct GNUNET_SCHEDULER_Task *task)
Cancel the task with the specified identifier.
Definition scheduler.c:980
struct GNUNET_PILS_Operation * op
The pils operation.

References GNUNET_CONTAINER_DLL_remove, GNUNET_free, GNUNET_notification_context_destroy(), GNUNET_PILS_cancel(), GNUNET_PILS_disconnect(), GNUNET_SCHEDULER_cancel(), GNUNET_TRANSPORT_core_disconnect(), nc, PilsRequest::op, pils, pils_requests_head, pils_requests_tail, rekey_task, and transport.

Referenced by GSC_KX_init(), and shutdown_task().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSC_NEIGHBOURS_check_excess_bandwidth()

int GSC_NEIGHBOURS_check_excess_bandwidth ( const struct GSC_KeyExchangeInfo target)

Check if the given neighbour has excess bandwidth available.

Parameters
targetneighbour to check
Returns
GNUNET_YES if excess bandwidth is available, GNUNET_NO if not

Definition at line 2982 of file gnunet-service-core_kx.c.

2983{
2984 return kxinfo->has_excess_bandwidth;
2985}

References GSC_KeyExchangeInfo::has_excess_bandwidth.

Referenced by try_transmission().

Here is the caller graph for this function:

◆ GSC_NEIGHBOURS_get_queue_length()

unsigned int GSC_NEIGHBOURS_get_queue_length ( const struct GSC_KeyExchangeInfo kxinfo)

Check how many messages are queued for the given neighbour.

Parameters
targetneighbour to check
Returns
number of items in the message queue
Parameters
kxinfodata about neighbour to check
Returns
number of items in the message queue

Definition at line 2975 of file gnunet-service-core_kx.c.

2976{
2977 return GNUNET_MQ_get_length (kxinfo->mq);
2978}
unsigned int GNUNET_MQ_get_length(struct GNUNET_MQ_Handle *mq)
Obtain the current length of the message queue.
Definition mq.c:293

References GNUNET_MQ_get_length(), and GSC_KeyExchangeInfo::mq.

Referenced by try_transmission().

Here is the call graph for this function:
Here is the caller graph for this function:

◆ GSC_KX_handle_client_monitor_peers()

void GSC_KX_handle_client_monitor_peers ( struct GNUNET_MQ_Handle mq)

Handle GNUNET_MESSAGE_TYPE_CORE_MONITOR_PEERS request.

For this request type, the client does not have to have transmitted an INIT request. All current peers are returned, regardless of which message types they accept.

Parameters
mqmessage queue to add for monitoring

Definition at line 2997 of file gnunet-service-core_kx.c.

2998{
2999 struct GNUNET_MQ_Envelope *env;
3000 struct MonitorNotifyMessage *done_msg;
3001 struct GSC_KeyExchangeInfo *kx;
3002
3004 for (kx = kx_head; NULL != kx; kx = kx->next)
3005 {
3006 struct GNUNET_MQ_Envelope *env_notify;
3007 struct MonitorNotifyMessage *msg;
3008
3010 msg->state = htonl ((uint32_t) kx->status);
3011 msg->peer = kx->peer;
3012 msg->timeout = GNUNET_TIME_absolute_hton (kx->timeout);
3013 GNUNET_MQ_send (mq, env_notify);
3014 }
3016 done_msg->state = htonl ((uint32_t) GNUNET_CORE_KX_ITERATION_FINISHED);
3019}
struct GNUNET_MessageHeader * msg
Definition 005.c:2
static struct GSC_KeyExchangeInfo * kx_head
DLL head.
@ GNUNET_CORE_KX_ITERATION_FINISHED
This is not a state in a peer's state machine, but a special value used with the GNUNET_CORE_MonitorC...
#define GNUNET_MQ_msg(mvar, type)
Allocate a GNUNET_MQ_Envelope.
void GNUNET_notification_context_add(struct GNUNET_NotificationContext *nc, struct GNUNET_MQ_Handle *mq)
Add a subscriber to the notification context.
Definition nc.c:161
#define GNUNET_MESSAGE_TYPE_CORE_MONITOR_NOTIFY
Reply for monitor by CORE service.
struct GNUNET_TIME_AbsoluteNBO GNUNET_TIME_absolute_hton(struct GNUNET_TIME_Absolute a)
Convert absolute time to network byte order.
Definition time.c:636
#define GNUNET_TIME_UNIT_FOREVER_ABS
Constant used to specify "forever".
static struct GNUNET_MQ_Handle * mq
Our connection to the resolver service, created on-demand, but then persists until error or shutdown.
Information about the status of a key exchange with another peer.
struct GSC_KeyExchangeInfo * next
DLL.
struct GNUNET_PeerIdentity peer
Identity of the peer.
struct GNUNET_TIME_Absolute timeout
When should the session time out (if there are no Acks to HEARTBEATs)?
enum GNUNET_CORE_KxState status
What is our connection state?
Message sent by the service to monitor clients to notify them about a peer changing status.
Definition core.h:313
uint32_t state
New peer state, an enum GNUNET_CORE_KxState in NBO.
Definition core.h:322
struct GNUNET_TIME_AbsoluteNBO timeout
How long will we stay in this state (if nothing else happens)?
Definition core.h:332

References env, GNUNET_CORE_KX_ITERATION_FINISHED, GNUNET_MESSAGE_TYPE_CORE_MONITOR_NOTIFY, GNUNET_MQ_msg, GNUNET_MQ_send(), GNUNET_notification_context_add(), GNUNET_TIME_absolute_hton(), GNUNET_TIME_UNIT_FOREVER_ABS, kx_head, mq, msg, nc, GSC_KeyExchangeInfo::next, GSC_KeyExchangeInfo::peer, MonitorNotifyMessage::state, GSC_KeyExchangeInfo::status, MonitorNotifyMessage::timeout, and GSC_KeyExchangeInfo::timeout.

Referenced by handle_client_monitor_peers().

Here is the call graph for this function:
Here is the caller graph for this function: