WIP: Layer 2-Implementierung #6
9 changed files with 349 additions and 42 deletions
|
@ -1,6 +1,9 @@
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
|
#define LOGGER_MODULE_NAME "conn"
|
||||||
|
#include "logger.h"
|
||||||
|
|
||||||
#include "connection.h"
|
#include "connection.h"
|
||||||
|
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
|
@ -11,7 +14,13 @@
|
||||||
|
|
||||||
#define SEQ_NR_MASK 0xF
|
#define SEQ_NR_MASK 0xF
|
||||||
|
|
||||||
result_t connection_init(connection_ctx_t *ctx, const ham64_t *my_addr, const ham64_t *peer_addr, connection_data_callback_t data_cb)
|
result_t connection_init(
|
||||||
|
connection_ctx_t *ctx,
|
||||||
|
const ham64_t *my_addr,
|
||||||
|
const ham64_t *peer_addr,
|
||||||
|
connection_data_callback_t data_cb,
|
||||||
|
connection_event_callback_t event_cb,
|
||||||
|
void *user_ctx)
|
||||||
{
|
{
|
||||||
ctx->last_acked_seq = 0;
|
ctx->last_acked_seq = 0;
|
||||||
ctx->next_expected_seq = 0;
|
ctx->next_expected_seq = 0;
|
||||||
|
@ -24,6 +33,13 @@ result_t connection_init(connection_ctx_t *ctx, const ham64_t *my_addr, const ha
|
||||||
ctx->peer_addr = *peer_addr;
|
ctx->peer_addr = *peer_addr;
|
||||||
|
|
||||||
ctx->data_cb = data_cb;
|
ctx->data_cb = data_cb;
|
||||||
|
ctx->event_cb = event_cb;
|
||||||
|
|
||||||
|
uint64_t now = get_hires_time();
|
||||||
|
ctx->last_rx_time = now;
|
||||||
|
ctx->retransmit_time = 0;
|
||||||
|
|
||||||
|
ctx->user_context = user_ctx;
|
||||||
|
|
||||||
ctx->conn_state = CONN_STATE_INITIALIZED;
|
ctx->conn_state = CONN_STATE_INITIALIZED;
|
||||||
|
|
||||||
|
@ -168,18 +184,15 @@ result_t connection_handle_packet_prechecked(
|
||||||
ctx->next_expected_seq++;
|
ctx->next_expected_seq++;
|
||||||
ctx->next_expected_seq &= 0xF;
|
ctx->next_expected_seq &= 0xF;
|
||||||
|
|
||||||
|
ctx->last_rx_time = get_hires_time();
|
||||||
|
|
||||||
LOG(LVL_INFO, "Received ACK for seq_nr %u in packet seq_nr %u.", header->rx_seq_nr, header->tx_seq_nr);
|
LOG(LVL_INFO, "Received ACK for seq_nr %u in packet seq_nr %u.", header->rx_seq_nr, header->tx_seq_nr);
|
||||||
|
|
||||||
// handle the acknowledgement internally
|
// handle the acknowledgement internally
|
||||||
connection_handle_ack(ctx, header->rx_seq_nr, true);
|
connection_handle_ack(ctx, header->rx_seq_nr, true);
|
||||||
|
|
||||||
size_t header_size = layer2_get_encoded_header_size(&header);
|
|
||||||
|
|
||||||
// extract the payload and forward it to the tun device
|
// extract the payload and forward it to the tun device
|
||||||
const uint8_t *payload = buf + header_size;
|
ctx->data_cb(ctx, payload, payload_len, ctx->user_context);
|
||||||
size_t payload_len = packet_size - header_size;
|
|
||||||
|
|
||||||
ctx->data_cb(ctx, payload, payload_len);
|
|
||||||
|
|
||||||
return OK;
|
return OK;
|
||||||
}
|
}
|
||||||
|
@ -287,7 +300,56 @@ result_t connection_add_empty_packet(connection_ctx_t *ctx, bool tx_request)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
size_t connection_encode_next_packet(connection_ctx_t *ctx, uint8_t ack_seq_nr, uint8_t *buf, size_t buf_len)
|
result_t connection_send_parameters(connection_ctx_t *ctx)
|
||||||
|
{
|
||||||
|
// check the connection state
|
||||||
|
switch(ctx->conn_state) {
|
||||||
|
case CONN_STATE_UNINITIALIZED:
|
||||||
|
case CONN_STATE_CLOSED:
|
||||||
|
case CONN_STATE_CONNECTING:
|
||||||
|
case CONN_STATE_ESTABLISHED:
|
||||||
|
LOG(LVL_ERR, "Trying to send connection parameters in state %u", ctx->conn_state);
|
||||||
|
return ERR_INVALID_STATE;
|
||||||
|
|
||||||
|
case CONN_STATE_INITIALIZED:
|
||||||
|
// in these states, packets can be handled
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
layer2_packet_header_t header;
|
||||||
|
|
||||||
|
header.dst_addr = ctx->peer_addr;
|
||||||
|
header.src_addr = ctx->my_addr;
|
||||||
|
header.msg_type = L2_MSG_TYPE_CONN_MGMT;
|
||||||
|
header.rx_seq_nr = 0; // will be filled in layer2_tx_encode_next_packet()
|
||||||
|
header.tx_seq_nr = ctx->next_seq_nr;
|
||||||
|
header.tx_request = 1;
|
||||||
|
|
||||||
|
size_t payload_len = 1;
|
||||||
|
uint8_t *payload = malloc(payload_len);
|
||||||
|
if(!payload) {
|
||||||
|
return ERR_NO_MEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
payload[0] = CONN_MGMT_TYPE_CONNECTION_PARAMETERS;
|
||||||
|
|
||||||
|
// TODO: calculate IP addresses from client’s HAM64 address
|
||||||
|
|
||||||
|
if (!packet_queue_add(&ctx->packet_queue, &header, payload, payload_len)) {
|
||||||
|
return ERR_NO_MEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx->next_seq_nr++;
|
||||||
|
ctx->next_seq_nr &= SEQ_NR_MASK;
|
||||||
|
|
||||||
|
// connection is considered established after the connection parameters are sent.
|
||||||
|
ctx->conn_state = CONN_STATE_ESTABLISHED;
|
||||||
|
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
size_t connection_encode_next_packet(connection_ctx_t *ctx, uint8_t *buf, size_t buf_len, bool *end_burst)
|
||||||
{
|
{
|
||||||
// check the connection state
|
// check the connection state
|
||||||
switch(ctx->conn_state) {
|
switch(ctx->conn_state) {
|
||||||
|
@ -311,7 +373,7 @@ size_t connection_encode_next_packet(connection_ctx_t *ctx, uint8_t ack_seq_nr,
|
||||||
}
|
}
|
||||||
|
|
||||||
layer2_packet_header_t header = entry->header;
|
layer2_packet_header_t header = entry->header;
|
||||||
header.rx_seq_nr = ack_seq_nr;
|
header.rx_seq_nr = ctx->next_expected_seq;
|
||||||
|
|
||||||
// encode the header
|
// encode the header
|
||||||
LOG(LVL_DEBUG, "Encoding packet with rx_seq_nr %u, tx_seq_nr %u.", header.rx_seq_nr, header.tx_seq_nr);
|
LOG(LVL_DEBUG, "Encoding packet with rx_seq_nr %u, tx_seq_nr %u.", header.rx_seq_nr, header.tx_seq_nr);
|
||||||
|
@ -324,6 +386,7 @@ size_t connection_encode_next_packet(connection_ctx_t *ctx, uint8_t ack_seq_nr,
|
||||||
|
|
||||||
ctx->next_packet_index++;
|
ctx->next_packet_index++;
|
||||||
|
|
||||||
|
*end_burst = header.tx_request;
|
||||||
return packet_size;
|
return packet_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -408,14 +471,37 @@ bool connection_can_transmit(const connection_ctx_t *ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool connection_is_closed(const connection_ctx_t *ctx)
|
||||||
|
{
|
||||||
|
switch(ctx->conn_state) {
|
||||||
|
case CONN_STATE_UNINITIALIZED:
|
||||||
|
case CONN_STATE_INITIALIZED:
|
||||||
|
case CONN_STATE_CLOSED:
|
||||||
|
return true;
|
||||||
|
|
||||||
|
case CONN_STATE_CONNECTING:
|
||||||
|
case CONN_STATE_ESTABLISHED:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
result_t connection_maintain(connection_ctx_t *ctx)
|
result_t connection_maintain(connection_ctx_t *ctx)
|
||||||
{
|
{
|
||||||
uint64_t now = get_hires_time();
|
uint64_t now = get_hires_time();
|
||||||
|
|
||||||
|
if(now > ctx->last_rx_time + HRTIME_MS(CONNECTION_TIMEOUT_MS)) {
|
||||||
|
LOG(LVL_INFO, "Connection timed out.");
|
||||||
|
ctx->conn_state = CONN_STATE_CLOSED;
|
||||||
|
ctx->event_cb(ctx, CONN_EVT_TIMEOUT, ctx->user_context);
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
if(ctx->retransmit_time != 0 && now >= ctx->retransmit_time) {
|
if(ctx->retransmit_time != 0 && now >= ctx->retransmit_time) {
|
||||||
LOG(LVL_INFO, "Retransmit triggered.");
|
LOG(LVL_INFO, "Retransmit triggered.");
|
||||||
ctx->retransmit_time = 0;
|
ctx->retransmit_time = 0;
|
||||||
connection_restart_tx(ctx);
|
connection_restart_tx(ctx);
|
||||||
|
ctx->event_cb(ctx, CONN_EVT_RETRANSMIT, ctx->user_context);
|
||||||
}
|
}
|
||||||
|
|
||||||
return OK;
|
return OK;
|
||||||
|
|
|
@ -25,12 +25,21 @@ typedef enum {
|
||||||
} connection_state_t;
|
} connection_state_t;
|
||||||
|
|
||||||
/*!\brief Type for a callback function that is called when a data packet was received. */
|
/*!\brief Type for a callback function that is called when a data packet was received. */
|
||||||
typedef void (*connection_data_callback_t)(struct connection_ctx_s *conn, const uint8_t *data, size_t len);
|
typedef void (*connection_data_callback_t)(struct connection_ctx_s *conn, const uint8_t *data, size_t len, void *user_ctx);
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
CONN_EVT_TIMEOUT, //!< The connection timed out because no packets were received
|
||||||
|
CONN_EVT_RETRANSMIT, //!< Packet queue transmission is restarted
|
||||||
|
} connection_evt_t;
|
||||||
|
|
||||||
|
/*!\brief Type for a callback function that is called on various connection events. */
|
||||||
|
typedef void (*connection_event_callback_t)(struct connection_ctx_s *conn, connection_evt_t evt, void *user_ctx);
|
||||||
|
|
||||||
typedef struct connection_ctx_s {
|
typedef struct connection_ctx_s {
|
||||||
connection_state_t conn_state; //!< State of the connection.
|
connection_state_t conn_state; //!< State of the connection.
|
||||||
|
|
||||||
connection_data_callback_t data_cb; //!< Callback function for received data packets.
|
connection_data_callback_t data_cb; //!< Callback function for received data packets.
|
||||||
|
connection_event_callback_t event_cb; //!< Callback function for event signalling.
|
||||||
|
|
||||||
ham64_t my_addr; //!< The local link layer address.
|
ham64_t my_addr; //!< The local link layer address.
|
||||||
ham64_t peer_addr; //!< The link layer address of the peer.
|
ham64_t peer_addr; //!< The link layer address of the peer.
|
||||||
|
@ -45,6 +54,8 @@ typedef struct connection_ctx_s {
|
||||||
|
|
||||||
uint64_t retransmit_time; //!< Time when a retransmit shall be triggered.
|
uint64_t retransmit_time; //!< Time when a retransmit shall be triggered.
|
||||||
uint64_t last_rx_time; //!< Time when a packet was last received and decoded.
|
uint64_t last_rx_time; //!< Time when a packet was last received and decoded.
|
||||||
|
|
||||||
|
void *user_context; //!< An arbitrary pointer that can set by the user.
|
||||||
} connection_ctx_t;
|
} connection_ctx_t;
|
||||||
|
|
||||||
|
|
||||||
|
@ -54,9 +65,17 @@ typedef struct connection_ctx_s {
|
||||||
* \param my_addr The local link layer address.
|
* \param my_addr The local link layer address.
|
||||||
* \param peer_addr The remote link layer address.
|
* \param peer_addr The remote link layer address.
|
||||||
* \param data_cb Callback for handling received payload data.
|
* \param data_cb Callback for handling received payload data.
|
||||||
|
* \param event_cb Callback for connection events.
|
||||||
|
* \param user_ctx User context pointer (for arbitrary data).
|
||||||
* \returns OK if everything worked or a fitting error code.
|
* \returns OK if everything worked or a fitting error code.
|
||||||
*/
|
*/
|
||||||
result_t connection_init(connection_ctx_t *ctx, const ham64_t *my_addr, const ham64_t *peer_addr, connection_data_callback_t data_cb);
|
result_t connection_init(
|
||||||
|
connection_ctx_t *ctx,
|
||||||
|
const ham64_t *my_addr,
|
||||||
|
const ham64_t *peer_addr,
|
||||||
|
connection_data_callback_t data_cb,
|
||||||
|
connection_event_callback_t event_cb,
|
||||||
|
void *user_ctx);
|
||||||
|
|
||||||
/*!\brief Destroy the given layer 2 connection context.
|
/*!\brief Destroy the given layer 2 connection context.
|
||||||
*/
|
*/
|
||||||
|
@ -113,6 +132,14 @@ bool connection_can_enqueue_packet(const connection_ctx_t *ctx);
|
||||||
*/
|
*/
|
||||||
result_t connection_add_empty_packet(connection_ctx_t *ctx, bool tx_request);
|
result_t connection_add_empty_packet(connection_ctx_t *ctx, bool tx_request);
|
||||||
|
|
||||||
|
/*!\brief Send connection parameters.
|
||||||
|
*
|
||||||
|
* This packet accepts an incoming connection request.
|
||||||
|
*
|
||||||
|
* \param ctx The connection context.
|
||||||
|
*/
|
||||||
|
result_t connection_send_parameters(connection_ctx_t *ctx);
|
||||||
|
|
||||||
/*!\brief Encode the next packet for transmission.
|
/*!\brief Encode the next packet for transmission.
|
||||||
*
|
*
|
||||||
* \note
|
* \note
|
||||||
|
@ -121,12 +148,12 @@ result_t connection_add_empty_packet(connection_ctx_t *ctx, bool tx_request);
|
||||||
* called to handle retransmits correctly.
|
* called to handle retransmits correctly.
|
||||||
*
|
*
|
||||||
* \param ctx The connection context.
|
* \param ctx The connection context.
|
||||||
* \param ack_seq_nr The received sequence number to send as an acknowledgement.
|
|
||||||
* \param buf Where to write the encoded packet data.
|
* \param buf Where to write the encoded packet data.
|
||||||
* \param buf_len Space available in the buffer.
|
* \param buf_len Space available in the buffer.
|
||||||
|
* \param end_burst Output parameter that is set to true if this is the last packet in a burst.
|
||||||
* \returns The number of bytes written to buf or zero if no packet was available.
|
* \returns The number of bytes written to buf or zero if no packet was available.
|
||||||
*/
|
*/
|
||||||
size_t connection_encode_next_packet(connection_ctx_t *ctx, uint8_t ack_seq_nr, uint8_t *buf, size_t buf_len);
|
size_t connection_encode_next_packet(connection_ctx_t *ctx, uint8_t *buf, size_t buf_len, bool *end_burst);
|
||||||
|
|
||||||
/*!\brief Restart the transmission from the beginning of the packet queue.
|
/*!\brief Restart the transmission from the beginning of the packet queue.
|
||||||
*/
|
*/
|
||||||
|
@ -150,6 +177,10 @@ void connection_handle_ack(connection_ctx_t *ctx, uint8_t acked_seq, bool do_ack
|
||||||
*/
|
*/
|
||||||
bool connection_can_transmit(const connection_ctx_t *ctx);
|
bool connection_can_transmit(const connection_ctx_t *ctx);
|
||||||
|
|
||||||
|
/*!\brief Check if this connection is closed.
|
||||||
|
*/
|
||||||
|
bool connection_is_closed(const connection_ctx_t *ctx);
|
||||||
|
|
||||||
/*!\brief Handle internal maintenance tasks.
|
/*!\brief Handle internal maintenance tasks.
|
||||||
*
|
*
|
||||||
* This should be called periodically to handle timeouts and retransmissions.
|
* This should be called periodically to handle timeouts and retransmissions.
|
||||||
|
|
|
@ -7,6 +7,12 @@
|
||||||
|
|
||||||
#define SEQ_NR_MASK 0xF
|
#define SEQ_NR_MASK 0xF
|
||||||
|
|
||||||
|
static destroy_entry(connection_list_entry_t *entry)
|
||||||
|
{
|
||||||
|
connection_destroy(&entry->connection);
|
||||||
|
free(entry);
|
||||||
|
}
|
||||||
|
|
||||||
// find the location where the timestamp should be inserted to maintain
|
// find the location where the timestamp should be inserted to maintain
|
||||||
// ascending timestamp order and return the then-previous entry.
|
// ascending timestamp order and return the then-previous entry.
|
||||||
//
|
//
|
||||||
|
@ -50,7 +56,7 @@ void connection_list_destroy(connection_list_t *list)
|
||||||
// delete all list entries
|
// delete all list entries
|
||||||
while(list->head) {
|
while(list->head) {
|
||||||
connection_list_entry_t *next = list->head->next;
|
connection_list_entry_t *next = list->head->next;
|
||||||
free(list->head);
|
destroy_entry(list->head);
|
||||||
list->head = next;
|
list->head = next;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -135,8 +141,39 @@ result_t connection_list_delete_head(connection_list_t *list)
|
||||||
|
|
||||||
connection_list_entry_t *new_head = list->head->next;
|
connection_list_entry_t *new_head = list->head->next;
|
||||||
|
|
||||||
free(list->head);
|
destroy_entry(list->head);
|
||||||
list->head = new_head;
|
list->head = new_head;
|
||||||
|
|
||||||
return OK;
|
return OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
result_t connection_list_delete_closed(connection_list_t *list)
|
||||||
|
{
|
||||||
|
if(!list->head) {
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
connection_list_entry_t *prev = NULL;
|
||||||
|
connection_list_entry_t *cur = list->head;
|
||||||
|
|
||||||
|
while(cur) {
|
||||||
|
if(connection_is_closed(&cur->connection)) {
|
||||||
|
connection_list_entry_t *to_delete = cur;
|
||||||
|
cur = cur->next;
|
||||||
|
|
||||||
|
if(prev) {
|
||||||
|
prev->next = to_delete->next;
|
||||||
|
} else {
|
||||||
|
list->head = to_delete->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
destroy_entry(to_delete);
|
||||||
|
} else {
|
||||||
|
prev = cur;
|
||||||
|
cur = cur->next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
|
@ -75,4 +75,12 @@ result_t connection_list_reschedule_head(connection_list_t *list, uint64_t next_
|
||||||
*/
|
*/
|
||||||
result_t connection_list_delete_head(connection_list_t *list);
|
result_t connection_list_delete_head(connection_list_t *list);
|
||||||
|
|
||||||
|
/*!\brief Delete all connections that are in a closed state.
|
||||||
|
*
|
||||||
|
* \param list Pointer to the list to clean.
|
||||||
|
*
|
||||||
|
* \returns OK if everything worked or a fitting error code.
|
||||||
|
*/
|
||||||
|
result_t connection_list_delete_closed(connection_list_t *list);
|
||||||
|
|
||||||
#endif // CONNECTION_LIST_H
|
#endif // CONNECTION_LIST_H
|
||||||
|
|
|
@ -1,14 +1,81 @@
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
|
#define LOGGER_MODULE_NAME "digi"
|
||||||
|
#include "logger.h"
|
||||||
|
|
||||||
#include "digipeater.h"
|
#include "digipeater.h"
|
||||||
|
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#include "layer2/connection.h"
|
#include "layer2/connection.h"
|
||||||
|
#include "layer2/connection_list.h"
|
||||||
#include "layer2/ham64.h"
|
#include "layer2/ham64.h"
|
||||||
|
#include "layer2/packet_structs.h"
|
||||||
#include "results.h"
|
#include "results.h"
|
||||||
#include "utils.h"
|
#include "utils.h"
|
||||||
|
|
||||||
|
void conn_data_cb(struct connection_ctx_s *conn, const uint8_t *data, size_t len, void *user_ctx)
|
||||||
|
{
|
||||||
|
(void)conn;
|
||||||
|
digipeater_ctx_t *digi_ctx = user_ctx;
|
||||||
|
|
||||||
|
digi_ctx->data_cb(digi_ctx, data, len);
|
||||||
|
}
|
||||||
|
|
||||||
|
void conn_event_cb(struct connection_ctx_s *conn, connection_evt_t evt, void *user_ctx)
|
||||||
|
{
|
||||||
|
(void)conn;
|
||||||
|
digipeater_ctx_t *digi_ctx = user_ctx;
|
||||||
|
|
||||||
|
switch(evt) {
|
||||||
|
case CONN_EVT_TIMEOUT:
|
||||||
|
// connection has been closed by timeout -> clean up the list
|
||||||
|
connection_list_delete_closed(&digi_ctx->conn_list);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
// do nothing
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static result_t digipeater_handle_beacon_responses(digipeater_ctx_t *ctx, const layer2_packet_header_t *header, const uint8_t *buf, size_t buf_len)
|
||||||
|
{
|
||||||
|
LOG(LVL_DEBUG, "Handling beacon response packet.");
|
||||||
|
layer2_dump_packet_header(LVL_DUMP, header);
|
||||||
|
|
||||||
|
if(header->msg_type != L2_MSG_TYPE_CONN_MGMT) {
|
||||||
|
LOG(LVL_ERR, "Beacon response with invalid message type %i", header->msg_type);
|
||||||
|
return ERR_INVALID_PARAM;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(buf_len == 0) {
|
||||||
|
LOG(LVL_ERR, "Missing payload in beacon response");
|
||||||
|
return ERR_INVALID_PARAM;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t conn_mgmt_type = buf[0];
|
||||||
|
|
||||||
|
if(conn_mgmt_type != CONN_MGMT_TYPE_CONNECTION_REQUEST) {
|
||||||
|
LOG(LVL_ERR, "Unexpected connection management type in beacon response: 0x%02x", conn_mgmt_type);
|
||||||
|
return ERR_INVALID_PARAM;
|
||||||
|
}
|
||||||
|
|
||||||
|
// packet is valid -> create a new connection, enqueue the parameters message
|
||||||
|
// and add it at the beginning of the connection list.
|
||||||
|
|
||||||
|
connection_ctx_t new_conn;
|
||||||
|
ERR_CHECK(connection_init(&new_conn, &ctx->my_addr, &header->src_addr, conn_data_cb, conn_event_cb, ctx));
|
||||||
|
|
||||||
|
ERR_CHECK(connection_send_parameters(&new_conn));
|
||||||
|
|
||||||
|
uint64_t now = get_hires_time();
|
||||||
|
ERR_CHECK(connection_list_insert(&ctx->conn_list, &new_conn, now));
|
||||||
|
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
result_t digipeater_init(digipeater_ctx_t *ctx, const ham64_t *my_addr, digipeater_data_callback_t data_cb)
|
result_t digipeater_init(digipeater_ctx_t *ctx, const ham64_t *my_addr, digipeater_data_callback_t data_cb)
|
||||||
{
|
{
|
||||||
ctx->my_addr = *my_addr;
|
ctx->my_addr = *my_addr;
|
||||||
|
@ -69,14 +136,28 @@ result_t digipeater_handle_packet(digipeater_ctx_t *ctx, const uint8_t *buf, siz
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: handle connection management packets here
|
// FIXME: handle connection management packets here
|
||||||
// FIXME: handle data and empty packets in the current connection
|
|
||||||
|
|
||||||
size_t header_size = layer2_get_encoded_header_size(&header);
|
size_t header_size = layer2_get_encoded_header_size(&header);
|
||||||
|
|
||||||
const uint8_t *payload = buf + header_size;
|
const uint8_t *payload = buf + header_size;
|
||||||
size_t payload_len = packet_size - header_size;
|
size_t payload_len = packet_size - header_size;
|
||||||
|
|
||||||
connection_handle_packet_prechecked(current_conn, &header, payload, payload_len);
|
switch(ctx->state) {
|
||||||
|
case DIGIPEATER_STATE_CONN:
|
||||||
|
{
|
||||||
|
connection_list_entry_t *head = connection_list_get_head(&ctx->conn_list);
|
||||||
|
if(head) {
|
||||||
|
connection_ctx_t *current_conn = &head->connection;
|
||||||
|
return connection_handle_packet_prechecked(current_conn, &header, payload, payload_len);
|
||||||
|
} else {
|
||||||
|
LOG(LVL_WARN, "Digipeater in CONN state, but there is no active connection! Packet dropped.");
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
case DIGIPEATER_STATE_BEACON:
|
||||||
|
return digipeater_handle_beacon_responses(ctx, &header, payload, payload_len);
|
||||||
|
}
|
||||||
|
|
||||||
return OK;
|
return OK;
|
||||||
}
|
}
|
||||||
|
@ -85,21 +166,50 @@ result_t digipeater_handle_packet(digipeater_ctx_t *ctx, const uint8_t *buf, siz
|
||||||
size_t digipeater_encode_next_packet(digipeater_ctx_t *ctx, uint8_t *buf, size_t buf_len, bool *end_burst)
|
size_t digipeater_encode_next_packet(digipeater_ctx_t *ctx, uint8_t *buf, size_t buf_len, bool *end_burst)
|
||||||
{
|
{
|
||||||
uint64_t now = get_hires_time();
|
uint64_t now = get_hires_time();
|
||||||
|
size_t packet_size = 0;
|
||||||
|
|
||||||
*end_burst = false;
|
*end_burst = false;
|
||||||
|
|
||||||
if(now > ctx->next_beacon_time) {
|
if(now > ctx->next_beacon_time) {
|
||||||
// TODO: build a beacon packet
|
// build a beacon packet
|
||||||
return 0;
|
layer2_packet_header_t header;
|
||||||
|
|
||||||
|
ham64_t broadcast = {{0xFFFF, 0, 0, 0}, 1};
|
||||||
|
|
||||||
|
header.dst_addr = broadcast;
|
||||||
|
header.src_addr = ctx->my_addr;
|
||||||
|
header.msg_type = L2_MSG_TYPE_CONN_MGMT;
|
||||||
|
header.rx_seq_nr = 0; // unused
|
||||||
|
header.tx_seq_nr = 0; // unused
|
||||||
|
header.tx_request = 1;
|
||||||
|
|
||||||
|
uint8_t payload[1] = {CONN_MGMT_TYPE_BEACON};
|
||||||
|
packet_size = layer2_encode_packet(&header, payload, 1, buf, buf_len);
|
||||||
|
|
||||||
|
ctx->state = DIGIPEATER_STATE_BEACON;
|
||||||
|
*end_burst = true;
|
||||||
|
return packet_size;
|
||||||
|
} else {
|
||||||
|
// pull packets from the current connection
|
||||||
|
|
||||||
|
connection_list_entry_t *head = connection_list_get_head(&ctx->conn_list);
|
||||||
|
if(!head) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
connection_ctx_t *conn = &head->connection;
|
||||||
|
if(connection_can_transmit(conn)) {
|
||||||
|
packet_size = connection_encode_next_packet(conn, buf, buf_len, end_burst);
|
||||||
|
|
||||||
|
ctx->state = DIGIPEATER_STATE_CONN;
|
||||||
|
}
|
||||||
|
|
||||||
|
return packet_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: pull packets from the current connection
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool digipeater_can_transmit(const digipeater_ctx_t *ctx)
|
bool digipeater_can_transmit(digipeater_ctx_t *ctx)
|
||||||
{
|
{
|
||||||
uint64_t now = get_hires_time();
|
uint64_t now = get_hires_time();
|
||||||
|
|
||||||
|
@ -107,16 +217,34 @@ bool digipeater_can_transmit(const digipeater_ctx_t *ctx)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: also true if next TX time has expired and there are packets waiting for this connection.
|
connection_list_entry_t *head = connection_list_get_head(&ctx->conn_list);
|
||||||
|
if(head) {
|
||||||
|
return connection_can_transmit(&head->connection);
|
||||||
|
}
|
||||||
|
|
||||||
/*return (packet_queue_get_used_space(&ctx->packet_queue) != 0)
|
|
||||||
&& (packet_queue_get(&ctx->packet_queue, ctx->next_packet_index) != NULL); */
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
result_t digipeater_maintain(digipeater_ctx_t *ctx)
|
result_t digipeater_maintain(digipeater_ctx_t *ctx)
|
||||||
{
|
{
|
||||||
(void)ctx;
|
switch(ctx->state) {
|
||||||
|
case DIGIPEATER_STATE_CONN:
|
||||||
|
{
|
||||||
|
connection_list_entry_t *head = connection_list_get_head(&ctx->conn_list);
|
||||||
|
if(!head) {
|
||||||
|
LOG(LVL_INFO, "No active connection -> force beacon state for packet handling.");
|
||||||
|
ctx->state = DIGIPEATER_STATE_BEACON;
|
||||||
|
} else {
|
||||||
|
ERR_CHECK(connection_maintain(&head->connection));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case DIGIPEATER_STATE_BEACON:
|
||||||
|
// nothing to do here
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
return OK;
|
return OK;
|
||||||
}
|
}
|
||||||
|
|
|
@ -83,7 +83,7 @@ size_t digipeater_encode_next_packet(digipeater_ctx_t *ctx, uint8_t *buf, size_t
|
||||||
|
|
||||||
/*!\brief Check if there are packets queued for transmission.
|
/*!\brief Check if there are packets queued for transmission.
|
||||||
*/
|
*/
|
||||||
bool digipeater_can_transmit(const digipeater_ctx_t *ctx);
|
bool digipeater_can_transmit(digipeater_ctx_t *ctx);
|
||||||
|
|
||||||
/*!\brief Handle internal maintenance tasks.
|
/*!\brief Handle internal maintenance tasks.
|
||||||
*
|
*
|
||||||
|
|
|
@ -33,6 +33,7 @@ typedef struct layer2_packet_header_s {
|
||||||
ham64_t dst_addr; //!< destination HAM-64 address
|
ham64_t dst_addr; //!< destination HAM-64 address
|
||||||
} layer2_packet_header_t;
|
} layer2_packet_header_t;
|
||||||
|
|
||||||
|
|
||||||
// maximum header size
|
// maximum header size
|
||||||
// - 1 byte packet info
|
// - 1 byte packet info
|
||||||
// - 1 byte sequence numbers
|
// - 1 byte sequence numbers
|
||||||
|
@ -104,6 +105,22 @@ typedef struct layer2_data_header_s {
|
||||||
|
|
||||||
/* Connection Management Structs */
|
/* Connection Management Structs */
|
||||||
|
|
||||||
// TODO
|
typedef enum {
|
||||||
|
CONN_MGMT_TYPE_BEACON = 0x00,
|
||||||
|
CONN_MGMT_TYPE_CONNECTION_REQUEST = 0x01,
|
||||||
|
CONN_MGMT_TYPE_CONNECTION_PARAMETERS = 0x02,
|
||||||
|
CONN_MGMT_TYPE_CONNECTION_RESET = 0x03,
|
||||||
|
CONN_MGMT_TYPE_DISCONNECT_REQUEST = 0x04,
|
||||||
|
CONN_MGMT_TYPE_DISCONNECT = 0x05,
|
||||||
|
} conn_mgmt_type_t;
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
CONN_PARAM_TYPE_IPV6_ADDRESS = 0x00,
|
||||||
|
CONN_PARAM_TYPE_IPV6_GATEWAY = 0x01,
|
||||||
|
CONN_PARAM_TYPE_IPV6_DNS = 0x02,
|
||||||
|
CONN_PARAM_TYPE_IPV4_ADDRESS = 0x08,
|
||||||
|
CONN_PARAM_TYPE_IPV4_GATEWAY = 0x09,
|
||||||
|
CONN_PARAM_TYPE_IPV4_DNS = 0x0A,
|
||||||
|
} conn_param_type_t;
|
||||||
|
|
||||||
#endif // PACKET_STRUCTS_H
|
#endif // PACKET_STRUCTS_H
|
||||||
|
|
|
@ -222,16 +222,13 @@ int main(int argc, char **argv)
|
||||||
size_t total_bytes = 0;
|
size_t total_bytes = 0;
|
||||||
uint64_t next_stats_print_time = old + HRTIME_MS(500);
|
uint64_t next_stats_print_time = old + HRTIME_MS(500);
|
||||||
|
|
||||||
uint64_t retransmit_time = 0;
|
// TODO: wait for beacon
|
||||||
|
// TODO: send connection request
|
||||||
|
|
||||||
while(m_running) {
|
while(m_running) {
|
||||||
uint64_t now = get_hires_time();
|
uint64_t now = get_hires_time();
|
||||||
|
|
||||||
if(retransmit_time != 0 && now >= retransmit_time) {
|
RESULT_CHECK(connection_maintain(&l2conn));
|
||||||
LOG(LVL_INFO, "Retransmit triggered.");
|
|
||||||
retransmit_time = 0;
|
|
||||||
connection_restart_tx(&l2conn);
|
|
||||||
}
|
|
||||||
|
|
||||||
// fill the TX queue from the TUN device
|
// fill the TX queue from the TUN device
|
||||||
while(connection_can_enqueue_packet(&l2conn)) {
|
while(connection_can_enqueue_packet(&l2conn)) {
|
||||||
|
|
|
@ -30,6 +30,7 @@
|
||||||
#include "debug_structs.h"
|
#include "debug_structs.h"
|
||||||
|
|
||||||
#include "layer2/connection.h"
|
#include "layer2/connection.h"
|
||||||
|
#include "layer2/connection_list.h"
|
||||||
|
|
||||||
#include "layer2/tundev.h"
|
#include "layer2/tundev.h"
|
||||||
|
|
||||||
|
@ -154,6 +155,9 @@ int main(int argc, char **argv)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
connection_list_t conn_list;
|
||||||
|
RESULT_CHECK(connection_list_init(&conn_list));
|
||||||
|
|
||||||
ham64_t my_address, peer_address;
|
ham64_t my_address, peer_address;
|
||||||
ham64_encode(MY_CALL, &my_address);
|
ham64_encode(MY_CALL, &my_address);
|
||||||
ham64_encode(PEER_CALL, &peer_address);
|
ham64_encode(PEER_CALL, &peer_address);
|
||||||
|
@ -222,18 +226,17 @@ int main(int argc, char **argv)
|
||||||
size_t total_bytes = 0;
|
size_t total_bytes = 0;
|
||||||
uint64_t next_stats_print_time = old + HRTIME_MS(500);
|
uint64_t next_stats_print_time = old + HRTIME_MS(500);
|
||||||
|
|
||||||
uint64_t retransmit_time = 0;
|
uint64_t next_beacon_time = old + HRTIME_MS(5000);
|
||||||
|
|
||||||
while(m_running) {
|
while(m_running) {
|
||||||
uint64_t now = get_hires_time();
|
uint64_t now = get_hires_time();
|
||||||
|
|
||||||
if(retransmit_time != 0 && now >= retransmit_time) {
|
if(now >= next_beacon_time) {
|
||||||
LOG(LVL_INFO, "Retransmit triggered.");
|
// TODO: encode and transmit beacon
|
||||||
retransmit_time = 0;
|
next_beacon_time += HRTIME_MS(5000);
|
||||||
connection_restart_tx(&l2conn);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// fill the TX queue from the TUN device
|
// FIXME: fill the TX queues from the TUN device
|
||||||
while(connection_can_enqueue_packet(&l2conn)) {
|
while(connection_can_enqueue_packet(&l2conn)) {
|
||||||
int ret = poll(&pfd_tun, 1, 0 /* timeout */);
|
int ret = poll(&pfd_tun, 1, 0 /* timeout */);
|
||||||
if(ret < 0) {
|
if(ret < 0) {
|
||||||
|
|
Loading…
Reference in a new issue