mirror of
https://git.sr.ht/~thestr4ng3r/chiaki
synced 2025-08-14 18:57:07 -07:00
Implement Congestion Control Thread
This commit is contained in:
parent
cb5bc75d2e
commit
4ea95a6130
5 changed files with 197 additions and 3 deletions
|
@ -22,7 +22,8 @@ set(HEADER_FILES
|
|||
include/chiaki/videoreceiver.h
|
||||
include/chiaki/frameprocessor.h
|
||||
include/chiaki/seqnum.h
|
||||
include/chiaki/discovery.h)
|
||||
include/chiaki/discovery.h
|
||||
include/chiaki/congestioncontrol.h)
|
||||
|
||||
set(SOURCE_FILES
|
||||
src/common.c
|
||||
|
@ -47,7 +48,8 @@ set(SOURCE_FILES
|
|||
src/audioreceiver.c
|
||||
src/videoreceiver.c
|
||||
src/frameprocessor.c
|
||||
src/discovery.c)
|
||||
src/discovery.c
|
||||
src/congestioncontrol.c)
|
||||
|
||||
add_subdirectory(protobuf)
|
||||
include_directories("${NANOPB_SOURCE_DIR}")
|
||||
|
|
44
lib/include/chiaki/congestioncontrol.h
Normal file
44
lib/include/chiaki/congestioncontrol.h
Normal file
|
@ -0,0 +1,44 @@
|
|||
/*
|
||||
* This file is part of Chiaki.
|
||||
*
|
||||
* Chiaki is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* Chiaki is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Chiaki. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#ifndef CHIAKI_CONGESTIONCONTROL_H
|
||||
#define CHIAKI_CONGESTIONCONTROL_H
|
||||
|
||||
#include "takion.h"
|
||||
#include "thread.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
typedef struct chiaki_congestion_control_t
|
||||
{
|
||||
ChiakiTakion *takion;
|
||||
ChiakiThread thread;
|
||||
ChiakiCond stop_cond;
|
||||
ChiakiMutex stop_cond_mutex;
|
||||
bool stop_cond_predicate;
|
||||
} ChiakiCongestionControl;
|
||||
|
||||
CHIAKI_EXPORT ChiakiErrorCode chiaki_congestion_control_start(ChiakiCongestionControl *control, ChiakiTakion *takion);
|
||||
CHIAKI_EXPORT ChiakiErrorCode chiaki_congestion_control_stop(ChiakiCongestionControl *control);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif // CHIAKI_CONGESTIONCONTROL_H
|
|
@ -58,6 +58,14 @@ typedef struct chiaki_takion_av_packet_t
|
|||
} ChiakiTakionAVPacket;
|
||||
|
||||
|
||||
typedef struct chiaki_takion_congestion_packet_t
|
||||
{
|
||||
uint16_t word_0;
|
||||
uint16_t word_1;
|
||||
uint16_t word_2;
|
||||
} ChiakiTakionCongestionPacket;
|
||||
|
||||
|
||||
typedef void (*ChiakiTakionDataCallback)(ChiakiTakionMessageDataType, uint8_t *buf, size_t buf_size, void *user);
|
||||
typedef void (*ChiakiTakionAVCallback)(ChiakiTakionAVPacket *packet, void *user);
|
||||
|
||||
|
@ -77,8 +85,13 @@ typedef struct chiaki_takion_connect_info_t
|
|||
typedef struct chiaki_takion_t
|
||||
{
|
||||
ChiakiLog *log;
|
||||
|
||||
ChiakiGKCrypt *gkcrypt_local; // if NULL (default), no gmac is calculated and nothing is encrypted
|
||||
size_t key_pos_local;
|
||||
ChiakiMutex gkcrypt_local_mutex;
|
||||
|
||||
ChiakiGKCrypt *gkcrypt_remote; // if NULL (default), remote gmacs are IGNORED (!) and everything is expected to be unencrypted
|
||||
|
||||
ChiakiTakionDataCallback data_cb;
|
||||
void *data_cb_user;
|
||||
ChiakiTakionAVCallback av_cb;
|
||||
|
@ -99,6 +112,7 @@ CHIAKI_EXPORT ChiakiErrorCode chiaki_takion_connect(ChiakiTakion *takion, Chiaki
|
|||
CHIAKI_EXPORT void chiaki_takion_close(ChiakiTakion *takion);
|
||||
CHIAKI_EXPORT ChiakiErrorCode chiaki_takion_send_raw(ChiakiTakion *takion, uint8_t *buf, size_t buf_size);
|
||||
CHIAKI_EXPORT ChiakiErrorCode chiaki_takion_send_message_data(ChiakiTakion *takion, uint32_t key_pos, uint8_t type_b, uint16_t channel, uint8_t *buf, size_t buf_size);
|
||||
CHIAKI_EXPORT ChiakiErrorCode chiaki_takion_send_congestion(ChiakiTakion *takion, ChiakiTakionCongestionPacket *packet);
|
||||
|
||||
static inline void chiaki_takion_set_crypt(ChiakiTakion *takion, ChiakiGKCrypt *gkcrypt_local, ChiakiGKCrypt *gkcrypt_remote) {
|
||||
takion->gkcrypt_local = gkcrypt_local;
|
||||
|
|
97
lib/src/congestioncontrol.c
Normal file
97
lib/src/congestioncontrol.c
Normal file
|
@ -0,0 +1,97 @@
|
|||
/*
|
||||
* This file is part of Chiaki.
|
||||
*
|
||||
* Chiaki is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* Chiaki is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Chiaki. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <chiaki/congestioncontrol.h>
|
||||
|
||||
|
||||
#define CONGESTION_CONTROL_INTERVAL 200 // ms
|
||||
|
||||
|
||||
static void *congestion_control_thread_func(void *user)
|
||||
{
|
||||
ChiakiCongestionControl *control = user;
|
||||
|
||||
ChiakiErrorCode err = chiaki_mutex_lock(&control->stop_cond_mutex);
|
||||
if(err != CHIAKI_ERR_SUCCESS)
|
||||
return NULL;
|
||||
|
||||
while(true)
|
||||
{
|
||||
err = chiaki_cond_timedwait(&control->stop_cond, &control->stop_cond_mutex, CONGESTION_CONTROL_INTERVAL);
|
||||
if(err != CHIAKI_ERR_SUCCESS && err != CHIAKI_ERR_TIMEOUT)
|
||||
break;
|
||||
if(control->stop_cond_predicate)
|
||||
break;
|
||||
// TODO: detect non-stop and non-timeout (spurious) wakeup and wait the rest of the timeout
|
||||
|
||||
CHIAKI_LOGD(control->takion->log, "Sending Congestion Control Packet\n");
|
||||
ChiakiTakionCongestionPacket packet = { 0 }; // TODO: fill with real values
|
||||
chiaki_takion_send_congestion(control->takion, &packet);
|
||||
}
|
||||
|
||||
chiaki_mutex_unlock(&control->stop_cond_mutex);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
CHIAKI_EXPORT ChiakiErrorCode chiaki_congestion_control_start(ChiakiCongestionControl *control, ChiakiTakion *takion)
|
||||
{
|
||||
control->takion = takion;
|
||||
|
||||
control->stop_cond_predicate = false;
|
||||
ChiakiErrorCode err = chiaki_cond_init(&control->stop_cond);
|
||||
if(err != CHIAKI_ERR_SUCCESS)
|
||||
return err;
|
||||
err = chiaki_mutex_init(&control->stop_cond_mutex);
|
||||
if(err != CHIAKI_ERR_SUCCESS)
|
||||
{
|
||||
chiaki_cond_fini(&control->stop_cond);
|
||||
return err;
|
||||
}
|
||||
|
||||
err = chiaki_thread_create(&control->thread, congestion_control_thread_func, control);
|
||||
if(err != CHIAKI_ERR_SUCCESS)
|
||||
{
|
||||
chiaki_mutex_fini(&control->stop_cond_mutex);
|
||||
chiaki_cond_fini(&control->stop_cond);
|
||||
return err;
|
||||
}
|
||||
|
||||
return CHIAKI_ERR_SUCCESS;
|
||||
}
|
||||
|
||||
CHIAKI_EXPORT ChiakiErrorCode chiaki_congestion_control_stop(ChiakiCongestionControl *control)
|
||||
{
|
||||
ChiakiErrorCode err = chiaki_mutex_lock(&control->stop_cond_mutex);
|
||||
if(err != CHIAKI_ERR_SUCCESS)
|
||||
return err;
|
||||
control->stop_cond_predicate = true;
|
||||
err = chiaki_cond_signal(&control->stop_cond);
|
||||
if(err != CHIAKI_ERR_SUCCESS)
|
||||
return err;
|
||||
err = chiaki_mutex_unlock(&control->stop_cond_mutex);
|
||||
if(err != CHIAKI_ERR_SUCCESS)
|
||||
return err;
|
||||
|
||||
err = chiaki_thread_join(&control->thread, NULL);
|
||||
if(err != CHIAKI_ERR_SUCCESS)
|
||||
return err;
|
||||
|
||||
chiaki_mutex_fini(&control->stop_cond_mutex);
|
||||
chiaki_cond_fini(&control->stop_cond);
|
||||
|
||||
return CHIAKI_ERR_SUCCESS;
|
||||
}
|
|
@ -16,6 +16,7 @@
|
|||
*/
|
||||
|
||||
#include <chiaki/takion.h>
|
||||
#include <chiaki/congestioncontrol.h>
|
||||
|
||||
#include <unistd.h>
|
||||
#include <fcntl.h>
|
||||
|
@ -26,6 +27,8 @@
|
|||
#include <assert.h>
|
||||
|
||||
|
||||
|
||||
|
||||
typedef enum takion_packet_type_t {
|
||||
TAKION_PACKET_TYPE_MESSAGE = 0,
|
||||
TAKION_PACKET_TYPE_VIDEO = 2,
|
||||
|
@ -101,10 +104,14 @@ static void takion_handle_packet_av(ChiakiTakion *takion, uint8_t base_type, uin
|
|||
|
||||
CHIAKI_EXPORT ChiakiErrorCode chiaki_takion_connect(ChiakiTakion *takion, ChiakiTakionConnectInfo *info)
|
||||
{
|
||||
ChiakiErrorCode ret;
|
||||
ChiakiErrorCode ret = CHIAKI_ERR_SUCCESS;
|
||||
|
||||
takion->log = info->log;
|
||||
takion->gkcrypt_local = NULL;
|
||||
ret = chiaki_mutex_init(&takion->gkcrypt_local_mutex);
|
||||
if(ret != CHIAKI_ERR_SUCCESS)
|
||||
return ret;
|
||||
takion->key_pos_local = 0;
|
||||
takion->gkcrypt_remote = NULL;
|
||||
takion->data_cb = info->data_cb;
|
||||
takion->data_cb_user = info->data_cb_user;
|
||||
|
@ -311,11 +318,38 @@ CHIAKI_EXPORT ChiakiErrorCode chiaki_takion_send_message_data_ack(ChiakiTakion *
|
|||
return chiaki_takion_send_raw(takion, buf, sizeof(buf));
|
||||
}
|
||||
|
||||
CHIAKI_EXPORT ChiakiErrorCode chiaki_takion_send_congestion(ChiakiTakion *takion, ChiakiTakionCongestionPacket *packet)
|
||||
{
|
||||
uint8_t buf[0xf];
|
||||
memset(buf, 0, sizeof(buf));
|
||||
buf[0] = 5;
|
||||
*((uint16_t *)(buf + 1)) = htons(packet->word_0);
|
||||
*((uint16_t *)(buf + 3)) = htons(packet->word_1);
|
||||
*((uint16_t *)(buf + 5)) = htons(packet->word_2);
|
||||
|
||||
ChiakiErrorCode err = chiaki_mutex_lock(&takion->gkcrypt_local_mutex);
|
||||
if(err != CHIAKI_ERR_SUCCESS)
|
||||
return err;
|
||||
*((uint32_t *)(buf + 0xb)) = htonl((uint32_t)takion->key_pos_local);
|
||||
err = chiaki_gkcrypt_gmac(takion->gkcrypt_local, takion->key_pos_local, buf, sizeof(buf), buf + 7);
|
||||
takion->key_pos_local += sizeof(buf);
|
||||
chiaki_mutex_unlock(&takion->gkcrypt_local_mutex);
|
||||
if(err != CHIAKI_ERR_SUCCESS)
|
||||
return err;
|
||||
|
||||
chiaki_log_hexdump(takion->log, CHIAKI_LOG_DEBUG, buf, sizeof(buf));
|
||||
|
||||
return chiaki_takion_send_raw(takion, buf, sizeof(buf));
|
||||
}
|
||||
|
||||
static void *takion_thread_func(void *user)
|
||||
{
|
||||
ChiakiTakion *takion = user;
|
||||
|
||||
ChiakiCongestionControl congestion_control;
|
||||
if(chiaki_congestion_control_start(&congestion_control, takion) != CHIAKI_ERR_SUCCESS)
|
||||
goto beach;
|
||||
|
||||
while(true)
|
||||
{
|
||||
uint8_t buf[1500];
|
||||
|
@ -326,6 +360,9 @@ static void *takion_thread_func(void *user)
|
|||
takion_handle_packet(takion, buf, received_size);
|
||||
}
|
||||
|
||||
chiaki_congestion_control_stop(&congestion_control);
|
||||
|
||||
beach:
|
||||
close(takion->sock);
|
||||
close(takion->stop_pipe[0]);
|
||||
return NULL;
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue