WIP: Layer 2-Implementierung #6
4 changed files with 196 additions and 40 deletions
|
@ -72,6 +72,18 @@ void connection_destroy(connection_ctx_t *ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void connection_setup_outgoing(connection_ctx_t *ctx)
|
||||||
|
{
|
||||||
|
ctx->conn_state = CONN_STATE_CONNECTING;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void connection_setup_incoming(connection_ctx_t *ctx)
|
||||||
|
{
|
||||||
|
ctx->conn_state = CONN_STATE_ESTABLISHED;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
result_t connection_handle_packet(connection_ctx_t *ctx, const uint8_t *buf, size_t buf_len, layer2_data_packet_t *data_packet)
|
result_t connection_handle_packet(connection_ctx_t *ctx, const uint8_t *buf, size_t buf_len, layer2_data_packet_t *data_packet)
|
||||||
{
|
{
|
||||||
// check the connection state
|
// check the connection state
|
||||||
|
@ -131,6 +143,48 @@ result_t connection_handle_packet(connection_ctx_t *ctx, const uint8_t *buf, siz
|
||||||
return connection_handle_packet_prechecked(ctx, &header, payload, payload_len, data_packet);
|
return connection_handle_packet_prechecked(ctx, &header, payload, payload_len, data_packet);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static result_t handle_conn_mgmt(
|
||||||
|
connection_ctx_t *ctx,
|
||||||
|
const layer2_packet_header_t *header,
|
||||||
|
const uint8_t *payload, size_t payload_len)
|
||||||
|
{
|
||||||
|
(void)header;
|
||||||
|
|
||||||
|
if(payload_len < 1) {
|
||||||
|
LOG(LVL_ERR, "Connection management packet without any payload is invalid.");
|
||||||
|
return ERR_INVALID_PARAM;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t packet_type = payload[0];
|
||||||
|
|
||||||
|
if(packet_type == CONN_MGMT_TYPE_BEACON) {
|
||||||
|
if(ctx->conn_state == CONN_STATE_CONNECTING) {
|
||||||
|
LOG(LVL_INFO, "Received beacon; queueing connection request.");
|
||||||
|
|
||||||
|
// enqueue a connection request packet
|
||||||
|
layer2_packet_header_t conn_request_header;
|
||||||
|
|
||||||
|
conn_request_header.tx_request = true;
|
||||||
|
conn_request_header.dst_addr = ctx->peer_addr;
|
||||||
|
conn_request_header.src_addr = ctx->my_addr;
|
||||||
|
conn_request_header.msg_type = L2_MSG_TYPE_CONN_MGMT;
|
||||||
|
conn_request_header.rx_seq_nr = 0;
|
||||||
|
conn_request_header.tx_seq_nr = 0;
|
||||||
|
|
||||||
|
// create a persistent copy of the packet data.
|
||||||
|
uint8_t packetbuf[1];
|
||||||
|
packetbuf[0] = CONN_MGMT_TYPE_CONNECTION_REQUEST;
|
||||||
|
|
||||||
|
connection_enqueue_packet(ctx, &conn_request_header, packetbuf, 1);
|
||||||
|
} else {
|
||||||
|
LOG(LVL_WARN, "Beacons are ignored in states other than CONNECTING.");
|
||||||
|
return ERR_INVALID_STATE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
result_t connection_handle_packet_prechecked(
|
result_t connection_handle_packet_prechecked(
|
||||||
connection_ctx_t *ctx,
|
connection_ctx_t *ctx,
|
||||||
const layer2_packet_header_t *header,
|
const layer2_packet_header_t *header,
|
||||||
|
@ -182,6 +236,8 @@ result_t connection_handle_packet_prechecked(
|
||||||
return OK; // do not ACK
|
return OK; // do not ACK
|
||||||
|
|
||||||
case L2_MSG_TYPE_CONN_MGMT:
|
case L2_MSG_TYPE_CONN_MGMT:
|
||||||
|
return handle_conn_mgmt(ctx, header, payload, payload_len);
|
||||||
|
|
||||||
case L2_MSG_TYPE_CONNECTIONLESS:
|
case L2_MSG_TYPE_CONNECTIONLESS:
|
||||||
LOG(LVL_WARN, "Message type %s is not implemented yet.", layer2_msg_type_to_string(header->msg_type));
|
LOG(LVL_WARN, "Message type %s is not implemented yet.", layer2_msg_type_to_string(header->msg_type));
|
||||||
return OK;
|
return OK;
|
||||||
|
@ -230,7 +286,39 @@ uint8_t connection_get_last_acked_seq(const connection_ctx_t *ctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
result_t connection_enqueue_packet(connection_ctx_t *ctx, uint8_t *buf, size_t buf_len)
|
result_t connection_enqueue_packet(
|
||||||
|
connection_ctx_t *ctx,
|
||||||
|
const layer2_packet_header_t *header,
|
||||||
|
uint8_t *payload, size_t payload_len)
|
||||||
|
{
|
||||||
|
if(packet_queue_get_free_space(&ctx->packet_queue) == 0) {
|
||||||
|
return ERR_NO_MEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t *packetbuf = NULL;
|
||||||
|
|
||||||
|
if(payload) {
|
||||||
|
// create a persistent copy of the packet data.
|
||||||
|
// TODO: possibly this copy operation can be removed by passing a malloc'd buffer in.
|
||||||
|
packetbuf = malloc(payload_len);
|
||||||
|
if(!packetbuf) {
|
||||||
|
LOG(LVL_ERR, "malloc failed.");
|
||||||
|
return ERR_NO_MEM;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(packetbuf, payload, payload_len);
|
||||||
|
|
||||||
|
packet_queue_add(&ctx->packet_queue, header, packetbuf, payload_len);
|
||||||
|
|
||||||
|
LOG(LVL_INFO, "Added packet tx_seq %u to queue -> %zu entries",
|
||||||
|
header->tx_seq_nr, packet_queue_get_used_space(&ctx->packet_queue));
|
||||||
|
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
result_t connection_enqueue_data_packet(connection_ctx_t *ctx, uint8_t *buf, size_t buf_len)
|
||||||
{
|
{
|
||||||
// check the connection state
|
// check the connection state
|
||||||
switch(ctx->conn_state) {
|
switch(ctx->conn_state) {
|
||||||
|
@ -248,10 +336,6 @@ result_t connection_enqueue_packet(connection_ctx_t *ctx, uint8_t *buf, size_t b
|
||||||
|
|
||||||
layer2_packet_header_t header;
|
layer2_packet_header_t header;
|
||||||
|
|
||||||
if(packet_queue_get_free_space(&ctx->packet_queue) == 0) {
|
|
||||||
return ERR_NO_MEM;
|
|
||||||
}
|
|
||||||
|
|
||||||
header.dst_addr = ctx->peer_addr;
|
header.dst_addr = ctx->peer_addr;
|
||||||
header.src_addr = ctx->my_addr;
|
header.src_addr = ctx->my_addr;
|
||||||
header.msg_type = L2_MSG_TYPE_DATA;
|
header.msg_type = L2_MSG_TYPE_DATA;
|
||||||
|
@ -259,20 +343,7 @@ result_t connection_enqueue_packet(connection_ctx_t *ctx, uint8_t *buf, size_t b
|
||||||
header.tx_request = 0;
|
header.tx_request = 0;
|
||||||
header.tx_seq_nr = ctx->next_seq_nr;
|
header.tx_seq_nr = ctx->next_seq_nr;
|
||||||
|
|
||||||
// create a persistent copy of the packet data.
|
ERR_CHECK(connection_enqueue_packet(ctx, &header, buf, buf_len));
|
||||||
// TODO: possibly this copy operation can be removed by passing a malloc'd buffer in.
|
|
||||||
uint8_t *packetbuf = malloc(buf_len);
|
|
||||||
if(!packetbuf) {
|
|
||||||
LOG(LVL_ERR, "malloc failed.");
|
|
||||||
return ERR_NO_MEM;
|
|
||||||
}
|
|
||||||
|
|
||||||
memcpy(packetbuf, buf, buf_len);
|
|
||||||
|
|
||||||
packet_queue_add(&ctx->packet_queue, &header, packetbuf, buf_len);
|
|
||||||
|
|
||||||
LOG(LVL_INFO, "Added packet tx_seq %u to queue -> %zu entries",
|
|
||||||
header.tx_seq_nr, packet_queue_get_used_space(&ctx->packet_queue));
|
|
||||||
|
|
||||||
ctx->next_seq_nr++;
|
ctx->next_seq_nr++;
|
||||||
ctx->next_seq_nr &= SEQ_NR_MASK;
|
ctx->next_seq_nr &= SEQ_NR_MASK;
|
||||||
|
@ -346,18 +417,13 @@ result_t connection_send_parameters(connection_ctx_t *ctx)
|
||||||
header.tx_request = 1;
|
header.tx_request = 1;
|
||||||
|
|
||||||
size_t payload_len = 1;
|
size_t payload_len = 1;
|
||||||
uint8_t *payload = malloc(payload_len);
|
uint8_t payload[payload_len];
|
||||||
if(!payload) {
|
|
||||||
return ERR_NO_MEM;
|
|
||||||
}
|
|
||||||
|
|
||||||
payload[0] = CONN_MGMT_TYPE_CONNECTION_PARAMETERS;
|
payload[0] = CONN_MGMT_TYPE_CONNECTION_PARAMETERS;
|
||||||
|
|
||||||
// TODO: calculate IP addresses from client’s HAM64 address
|
// TODO: calculate IP addresses from client’s HAM64 address
|
||||||
|
|
||||||
if (!packet_queue_add(&ctx->packet_queue, &header, payload, payload_len)) {
|
ERR_CHECK(connection_enqueue_packet(ctx, &header, payload, payload_len));
|
||||||
return ERR_NO_MEM;
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx->next_seq_nr++;
|
ctx->next_seq_nr++;
|
||||||
ctx->next_seq_nr &= SEQ_NR_MASK;
|
ctx->next_seq_nr &= SEQ_NR_MASK;
|
||||||
|
|
|
@ -69,6 +69,20 @@ result_t connection_init(
|
||||||
*/
|
*/
|
||||||
void connection_destroy(connection_ctx_t *ctx);
|
void connection_destroy(connection_ctx_t *ctx);
|
||||||
|
|
||||||
|
/*!\brief Set up a outgoing connection.
|
||||||
|
*
|
||||||
|
* This puts the connection into CONN_STATE_CONNECTING state, which causes
|
||||||
|
* beacons to be handled.
|
||||||
|
*/
|
||||||
|
void connection_setup_outgoing(connection_ctx_t *ctx);
|
||||||
|
|
||||||
|
/*!\brief Set up an incoming connection.
|
||||||
|
*
|
||||||
|
* As this function is intended to be called after a connection request was
|
||||||
|
* handled, it puts the connection directly into CONN_STATE_ESTABLISHED state.
|
||||||
|
*/
|
||||||
|
void connection_setup_incoming(connection_ctx_t *ctx);
|
||||||
|
|
||||||
/*!\brief Handle a received packet.
|
/*!\brief Handle a received packet.
|
||||||
*
|
*
|
||||||
* \param[inout] ctx The connection context.
|
* \param[inout] ctx The connection context.
|
||||||
|
@ -112,9 +126,22 @@ uint8_t connection_get_next_expected_seq(const connection_ctx_t *ctx);
|
||||||
uint8_t connection_get_last_acked_seq(const connection_ctx_t *ctx);
|
uint8_t connection_get_last_acked_seq(const connection_ctx_t *ctx);
|
||||||
|
|
||||||
/*!\brief Enqueue a packet for transmission.
|
/*!\brief Enqueue a packet for transmission.
|
||||||
* \param ctx The connection context.
|
* \param ctx The connection context.
|
||||||
|
* \param header Pointer to the packet header.
|
||||||
|
* \param payload Pointer to the payload.
|
||||||
|
* \param payload_len Length of the payload.
|
||||||
*/
|
*/
|
||||||
result_t connection_enqueue_packet(connection_ctx_t *ctx, uint8_t *buf, size_t buf_len);
|
result_t connection_enqueue_packet(
|
||||||
|
connection_ctx_t *ctx,
|
||||||
|
const layer2_packet_header_t *header,
|
||||||
|
uint8_t *payload, size_t payload_len);
|
||||||
|
|
||||||
|
/*!\brief Enqueue a data packet for transmission.
|
||||||
|
* \param ctx The connection context.
|
||||||
|
* \param buf Pointer to the data buffer.
|
||||||
|
* \param buf_len Length of the data.
|
||||||
|
*/
|
||||||
|
result_t connection_enqueue_data_packet(connection_ctx_t *ctx, uint8_t *buf, size_t buf_len);
|
||||||
|
|
||||||
/*!\brief Check if there is free space in the TX packet queue.
|
/*!\brief Check if there is free space in the TX packet queue.
|
||||||
* \param ctx The connection context.
|
* \param ctx The connection context.
|
||||||
|
|
|
@ -213,7 +213,7 @@ result_t connection_list_enqueue_packet(connection_list_t *list, uint8_t *data,
|
||||||
return ERR_INVALID_ADDRESS;
|
return ERR_INVALID_ADDRESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
return connection_enqueue_packet(&ptr->connection, data, data_len);
|
return connection_enqueue_data_packet(&ptr->connection, data, data_len);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -122,6 +122,71 @@ void rx_data_to_tun(const layer2_data_packet_t *data_packet)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
result_t connect_to_digipeater(connection_ctx_t *conn)
|
||||||
|
{
|
||||||
|
while(conn->conn_state != CONN_STATE_ESTABLISHED) {
|
||||||
|
uint8_t packetbuf[65536];
|
||||||
|
|
||||||
|
LOG(LVL_INFO, "Waiting for packets from digipeater (conn. state: %d)", conn->conn_state);
|
||||||
|
|
||||||
|
// note: recv() is blocking here
|
||||||
|
int ret = recv(m_bcast_sock, packetbuf, sizeof(packetbuf), 0);
|
||||||
|
if(ret < 0) {
|
||||||
|
LOG(LVL_ERR, "recv: %s", strerror(errno));
|
||||||
|
return ERR_SYSCALL;
|
||||||
|
} else if(ret == 0) {
|
||||||
|
LOG(LVL_ERR, "recv() returned zero.");
|
||||||
|
return ERR_SYSCALL;
|
||||||
|
}
|
||||||
|
|
||||||
|
layer2_data_packet_t data_packet;
|
||||||
|
result_t err_code = connection_handle_packet(conn, packetbuf, ret, &data_packet);
|
||||||
|
|
||||||
|
switch(err_code) {
|
||||||
|
case OK:
|
||||||
|
break; // state machine advanced and there are probably packets to send
|
||||||
|
|
||||||
|
case ERR_INTEGRITY:
|
||||||
|
case ERR_INVALID_STATE:
|
||||||
|
continue; // retry with the next packet
|
||||||
|
|
||||||
|
default:
|
||||||
|
LOG(LVL_ERR, "connection_handle_packet() returned %d.", err_code);
|
||||||
|
return err_code;
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG(LVL_INFO, "Packet processed successfully (new conn. state: %d); Transmitting response (if any).", conn->conn_state);
|
||||||
|
|
||||||
|
// send any packets in the queue
|
||||||
|
while(connection_can_transmit(conn)) {
|
||||||
|
uint8_t packet_buf[2048];
|
||||||
|
size_t packet_size;
|
||||||
|
bool end_burst;
|
||||||
|
|
||||||
|
packet_size = connection_encode_next_packet(conn,
|
||||||
|
packet_buf, sizeof(packet_buf), &end_burst);
|
||||||
|
|
||||||
|
if(packet_size == 0) {
|
||||||
|
// no more packets available
|
||||||
|
LOG(LVL_DEBUG, "Ending burst due to empty packet queue.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG(LVL_DEBUG, "Adding packet with %d bytes to burst.", packet_size);
|
||||||
|
|
||||||
|
RESULT_CHECK(transmit(packet_buf, packet_size));
|
||||||
|
|
||||||
|
if(end_burst) {
|
||||||
|
LOG(LVL_DEBUG, "Ending burst on request.");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void conn_evt_cb(struct connection_ctx_s *conn, connection_evt_t evt, void *user_ctx)
|
void conn_evt_cb(struct connection_ctx_s *conn, connection_evt_t evt, void *user_ctx)
|
||||||
{
|
{
|
||||||
(void)conn;
|
(void)conn;
|
||||||
|
@ -151,14 +216,6 @@ int main(void)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
ham64_t my_address, peer_address;
|
|
||||||
ham64_encode(MY_CALL, &my_address);
|
|
||||||
ham64_encode(PEER_CALL, &peer_address);
|
|
||||||
RESULT_CHECK(connection_init(&l2conn, &my_address, &peer_address));
|
|
||||||
|
|
||||||
// force connection into the established state
|
|
||||||
l2conn.conn_state = CONN_STATE_ESTABLISHED;
|
|
||||||
|
|
||||||
// ** Set up signal handling
|
// ** Set up signal handling
|
||||||
|
|
||||||
struct sigaction term_action = {0};
|
struct sigaction term_action = {0};
|
||||||
|
@ -219,8 +276,14 @@ int main(void)
|
||||||
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);
|
||||||
|
|
||||||
// TODO: wait for beacon
|
ham64_t my_address, peer_address;
|
||||||
// TODO: send connection request
|
ham64_encode(MY_CALL, &my_address);
|
||||||
|
ham64_encode(PEER_CALL, &peer_address);
|
||||||
|
RESULT_CHECK(connection_init(&l2conn, &my_address, &peer_address));
|
||||||
|
|
||||||
|
connection_setup_outgoing(&l2conn);
|
||||||
|
|
||||||
|
RESULT_CHECK(connect_to_digipeater(&l2conn));
|
||||||
|
|
||||||
while(m_running) {
|
while(m_running) {
|
||||||
connection_evt_t evt;
|
connection_evt_t evt;
|
||||||
|
@ -256,7 +319,7 @@ int main(void)
|
||||||
LOG(LVL_DUMP, "TUN Flags: 0x%04x", *(uint16_t*)packetbuf);
|
LOG(LVL_DUMP, "TUN Flags: 0x%04x", *(uint16_t*)packetbuf);
|
||||||
LOG(LVL_DUMP, "TUN Proto: 0x%04x", *((uint16_t*)packetbuf + 1));
|
LOG(LVL_DUMP, "TUN Proto: 0x%04x", *((uint16_t*)packetbuf + 1));
|
||||||
|
|
||||||
RESULT_CHECK(connection_enqueue_packet(&l2conn, packetbuf, ret));
|
RESULT_CHECK(connection_enqueue_data_packet(&l2conn, packetbuf, ret));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue