2023-11-15 23:46:08 -04:00
/*
class for networking mapped ports
*/
# include "AP_Networking_Config.h"
2024-01-24 00:43:53 -04:00
# if AP_NETWORKING_REGISTER_PORT_ENABLED
2023-11-15 23:46:08 -04:00
# include "AP_Networking.h"
# include <AP_HAL/utility/Socket.h>
# include <GCS_MAVLink/GCS.h>
# include <AP_BoardConfig/AP_BoardConfig.h>
# include <AP_Math/AP_Math.h>
2023-11-25 20:46:07 -04:00
# include <AP_SerialManager/AP_SerialManager.h>
# include <AP_HAL/utility/packetise.h>
2023-11-15 23:46:08 -04:00
extern const AP_HAL : : HAL & hal ;
2023-11-18 17:46:46 -04:00
# ifndef AP_NETWORKING_PORT_MIN_TXSIZE
# define AP_NETWORKING_PORT_MIN_TXSIZE 2048
# endif
# ifndef AP_NETWORKING_PORT_MIN_RXSIZE
# define AP_NETWORKING_PORT_MIN_RXSIZE 2048
# endif
2023-11-25 20:46:07 -04:00
# ifndef AP_NETWORKING_PORT_STACK_SIZE
# define AP_NETWORKING_PORT_STACK_SIZE 1024
# endif
2023-11-15 23:46:08 -04:00
const AP_Param : : GroupInfo AP_Networking : : Port : : var_info [ ] = {
// @Param: TYPE
// @DisplayName: Port type
2023-12-02 18:08:43 -04:00
// @Description: Port type for network serial port. For the two client types a valid destination IP address must be set. For the two server types either 0.0.0.0 or a local address can be used. The UDP client type will use broadcast if the IP is set to 255.255.255.255 and will use UDP multicast if the IP is in the multicast address range.
// @Values: 0:Disabled, 1:UDP client, 2:UDP server, 3:TCP client, 4:TCP server
2023-11-15 23:46:08 -04:00
// @RebootRequired: True
// @User: Advanced
AP_GROUPINFO_FLAGS ( " TYPE " , 1 , AP_Networking : : Port , type , 0 , AP_PARAM_FLAG_ENABLE ) ,
// @Param: PROTOCOL
2024-01-05 17:25:07 -04:00
// @DisplayName: Protocol
// @Description: Networked serial port protocol
2023-11-15 23:46:08 -04:00
// @User: Advanced
2024-01-05 17:25:07 -04:00
// @RebootRequired: True
2023-12-02 18:08:43 -04:00
// @CopyFieldsFrom: SERIAL1_PROTOCOL
2023-11-15 23:46:08 -04:00
AP_GROUPINFO ( " PROTOCOL " , 2 , AP_Networking : : Port , state . protocol , 0 ) ,
// @Group: IP
// @Path: AP_Networking_address.cpp
2024-01-05 17:25:07 -04:00
// @RebootRequired : True
2023-11-15 23:46:08 -04:00
AP_SUBGROUPINFO ( ip , " IP " , 3 , AP_Networking : : Port , AP_Networking_IPV4 ) ,
// @Param: PORT
// @DisplayName: Port number
// @Description: Port number
// @Range: 0 65535
// @RebootRequired: True
// @User: Advanced
AP_GROUPINFO ( " PORT " , 4 , AP_Networking : : Port , port , 0 ) ,
AP_GROUPEND
} ;
/*
initialise mapped network ports
*/
void AP_Networking : : ports_init ( void )
{
2023-11-21 20:33:48 -04:00
for ( uint8_t i = 0 ; i < ARRAY_SIZE ( ports ) ; i + + ) {
2023-11-15 23:46:08 -04:00
auto & p = ports [ i ] ;
NetworkPortType ptype = ( NetworkPortType ) p . type ;
p . state . idx = AP_SERIALMANAGER_NET_PORT_1 + i ;
switch ( ptype ) {
case NetworkPortType : : NONE :
break ;
case NetworkPortType : : UDP_CLIENT :
2023-11-25 20:46:07 -04:00
p . udp_client_init ( ) ;
break ;
case NetworkPortType : : UDP_SERVER :
p . udp_server_init ( ) ;
break ;
case NetworkPortType : : TCP_SERVER :
p . tcp_server_init ( ) ;
break ;
case NetworkPortType : : TCP_CLIENT :
p . tcp_client_init ( ) ;
2023-11-15 23:46:08 -04:00
break ;
}
2023-11-25 20:46:07 -04:00
if ( p . sock ! = nullptr | | p . listen_sock ! = nullptr ) {
2023-11-15 23:46:08 -04:00
AP : : serialmanager ( ) . register_port ( & p ) ;
}
}
}
/*
2023-11-25 20:46:07 -04:00
wrapper for thread_create for port functions
2023-11-15 23:46:08 -04:00
*/
2023-11-25 20:46:07 -04:00
void AP_Networking : : Port : : thread_create ( AP_HAL : : MemberProc proc )
2023-11-15 23:46:08 -04:00
{
2023-11-25 20:46:07 -04:00
const uint8_t idx = state . idx - AP_SERIALMANAGER_NET_PORT_1 ;
hal . util - > snprintf ( thread_name , sizeof ( thread_name ) , " NET_P%u " , unsigned ( idx ) ) ;
if ( ! init_buffers ( AP_NETWORKING_PORT_MIN_RXSIZE , AP_NETWORKING_PORT_MIN_TXSIZE ) ) {
AP_BoardConfig : : allocation_error ( " Failed to allocate %s buffers " , thread_name ) ;
2023-11-15 23:46:08 -04:00
return ;
}
2023-11-25 20:46:07 -04:00
if ( ! hal . scheduler - > thread_create ( proc , thread_name , AP_NETWORKING_PORT_STACK_SIZE , AP_HAL : : Scheduler : : PRIORITY_UART , 0 ) ) {
AP_BoardConfig : : allocation_error ( " Failed to allocate %s client thread " , thread_name ) ;
}
}
/*
initialise a UDP client
*/
void AP_Networking : : Port : : udp_client_init ( void )
{
sock = new SocketAPM ( true ) ;
if ( sock = = nullptr ) {
2023-11-17 18:26:11 -04:00
return ;
2023-11-17 15:21:12 -04:00
}
2023-11-25 20:46:07 -04:00
sock - > set_blocking ( false ) ;
// setup for packet boundaries if this is mavlink
packetise = ( state . protocol = = AP_SerialManager : : SerialProtocol_MAVLink | |
state . protocol = = AP_SerialManager : : SerialProtocol_MAVLink2 ) ;
thread_create ( FUNCTOR_BIND_MEMBER ( & AP_Networking : : Port : : udp_client_loop , void ) ) ;
}
/*
initialise a UDP server
*/
void AP_Networking : : Port : : udp_server_init ( void )
{
2023-11-15 23:46:08 -04:00
sock = new SocketAPM ( true ) ;
if ( sock = = nullptr ) {
return ;
}
2023-11-25 20:46:07 -04:00
sock - > set_blocking ( false ) ;
// setup for packet boundaries if this is mavlink
packetise = ( state . protocol = = AP_SerialManager : : SerialProtocol_MAVLink | |
state . protocol = = AP_SerialManager : : SerialProtocol_MAVLink2 ) ;
thread_create ( FUNCTOR_BIND_MEMBER ( & AP_Networking : : Port : : udp_server_loop , void ) ) ;
}
/*
initialise a TCP server
*/
void AP_Networking : : Port : : tcp_server_init ( void )
{
listen_sock = new SocketAPM ( false ) ;
if ( listen_sock = = nullptr ) {
return ;
2023-11-15 23:46:08 -04:00
}
2023-11-25 20:46:07 -04:00
listen_sock - > reuseaddress ( ) ;
thread_create ( FUNCTOR_BIND_MEMBER ( & AP_Networking : : Port : : tcp_server_loop , void ) ) ;
2023-11-15 23:46:08 -04:00
}
/*
2023-11-25 20:46:07 -04:00
initialise a TCP client
2023-11-15 23:46:08 -04:00
*/
2023-11-25 20:46:07 -04:00
void AP_Networking : : Port : : tcp_client_init ( void )
{
sock = new SocketAPM ( false ) ;
if ( sock ! = nullptr ) {
sock - > set_blocking ( true ) ;
thread_create ( FUNCTOR_BIND_MEMBER ( & AP_Networking : : Port : : tcp_client_loop , void ) ) ;
}
}
/*
update a UDP client
*/
void AP_Networking : : Port : : udp_client_loop ( void )
{
2023-11-29 18:23:39 -04:00
AP : : network ( ) . startup_wait ( ) ;
2023-11-15 23:46:08 -04:00
const char * dest = ip . get_str ( ) ;
if ( ! sock - > connect ( dest , port . get ( ) ) ) {
GCS_SEND_TEXT ( MAV_SEVERITY_ERROR , " UDP[%u]: Failed to connect to %s " , ( unsigned ) state . idx , dest ) ;
delete sock ;
sock = nullptr ;
return ;
}
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " UDP[%u]: connected to %s:%u " , ( unsigned ) state . idx , dest , unsigned ( port . get ( ) ) ) ;
2023-11-25 20:46:07 -04:00
connected = true ;
bool active = false ;
2023-11-15 23:46:08 -04:00
while ( true ) {
2023-11-25 20:46:07 -04:00
if ( ! active ) {
hal . scheduler - > delay_microseconds ( 100 ) ;
}
active = send_receive ( ) ;
}
}
2023-11-15 23:46:08 -04:00
2023-11-25 20:46:07 -04:00
/*
update a UDP server
*/
void AP_Networking : : Port : : udp_server_loop ( void )
{
2023-11-29 18:23:39 -04:00
AP : : network ( ) . startup_wait ( ) ;
2023-11-25 20:46:07 -04:00
const char * addr = ip . get_str ( ) ;
if ( ! sock - > bind ( addr , port . get ( ) ) ) {
GCS_SEND_TEXT ( MAV_SEVERITY_ERROR , " UDP[%u]: Failed to bind to %s:%u " , ( unsigned ) state . idx , addr , unsigned ( port . get ( ) ) ) ;
delete sock ;
sock = nullptr ;
return ;
}
sock - > reuseaddress ( ) ;
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " UDP[%u]: bound to %s:%u " , ( unsigned ) state . idx , addr , unsigned ( port . get ( ) ) ) ;
bool active = false ;
while ( true ) {
if ( ! active ) {
hal . scheduler - > delay_microseconds ( 100 ) ;
}
active = send_receive ( ) ;
}
}
/*
update a TCP server
*/
void AP_Networking : : Port : : tcp_server_loop ( void )
{
2023-11-29 18:23:39 -04:00
AP : : network ( ) . startup_wait ( ) ;
2023-11-25 20:46:07 -04:00
const char * addr = ip . get_str ( ) ;
if ( ! listen_sock - > bind ( addr , port . get ( ) ) | | ! listen_sock - > listen ( 1 ) ) {
GCS_SEND_TEXT ( MAV_SEVERITY_ERROR , " TCP[%u]: Failed to bind to %s:%u " , ( unsigned ) state . idx , addr , unsigned ( port . get ( ) ) ) ;
delete listen_sock ;
listen_sock = nullptr ;
return ;
}
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " TCP[%u]: bound to %s:%u " , ( unsigned ) state . idx , addr , unsigned ( port . get ( ) ) ) ;
close_on_recv_error = true ;
bool active = false ;
while ( true ) {
if ( ! active ) {
hal . scheduler - > delay_microseconds ( 100 ) ;
}
if ( sock = = nullptr ) {
sock = listen_sock - > accept ( 100 ) ;
if ( sock ! = nullptr ) {
sock - > set_blocking ( false ) ;
char buf [ 16 ] ;
uint16_t last_port ;
const char * last_addr = listen_sock - > last_recv_address ( buf , sizeof ( buf ) , last_port ) ;
if ( last_addr ! = nullptr ) {
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " TCP[%u]: connection from %s:%u " , ( unsigned ) state . idx , last_addr , unsigned ( last_port ) ) ;
}
connected = true ;
sock - > reuseaddress ( ) ;
2023-11-15 23:46:08 -04:00
}
}
2023-11-25 20:46:07 -04:00
if ( sock ! = nullptr ) {
active = send_receive ( ) ;
}
}
}
2023-11-15 23:46:08 -04:00
2023-11-25 20:46:07 -04:00
/*
update a TCP client
*/
void AP_Networking : : Port : : tcp_client_loop ( void )
{
2023-11-29 18:23:39 -04:00
AP : : network ( ) . startup_wait ( ) ;
2023-11-25 20:46:07 -04:00
close_on_recv_error = true ;
bool active = false ;
while ( true ) {
if ( ! active ) {
hal . scheduler - > delay_microseconds ( 100 ) ;
}
if ( sock = = nullptr ) {
sock = new SocketAPM ( false ) ;
if ( sock = = nullptr ) {
continue ;
}
sock - > set_blocking ( true ) ;
connected = false ;
}
if ( ! connected ) {
const char * dest = ip . get_str ( ) ;
connected = sock - > connect ( dest , port . get ( ) ) ;
if ( connected ) {
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " TCP[%u]: connected to %s:%u " , unsigned ( state . idx ) , dest , unsigned ( port . get ( ) ) ) ;
sock - > set_blocking ( false ) ;
} else {
delete sock ;
sock = nullptr ;
// don't try and connect too fast
hal . scheduler - > delay ( 100 ) ;
}
}
if ( sock ! = nullptr & & connected ) {
active = send_receive ( ) ;
}
}
}
/*
run one send / receive loop
*/
bool AP_Networking : : Port : : send_receive ( void )
{
bool active = false ;
2024-01-09 16:36:47 -04:00
uint32_t space ;
2023-11-25 20:46:07 -04:00
// handle incoming packets
2024-01-09 16:36:47 -04:00
{
WITH_SEMAPHORE ( sem ) ;
space = readbuffer - > space ( ) ;
}
2023-11-25 20:46:07 -04:00
if ( space > 0 ) {
const uint32_t n = MIN ( 300U , space ) ;
uint8_t buf [ n ] ;
const auto ret = sock - > recv ( buf , n , 0 ) ;
if ( close_on_recv_error & & ret = = 0 ) {
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " TCP[%u]: closed connection " , unsigned ( state . idx ) ) ;
delete sock ;
sock = nullptr ;
return false ;
}
if ( ret > 0 ) {
2024-01-09 16:36:47 -04:00
WITH_SEMAPHORE ( sem ) ;
2023-11-25 20:46:07 -04:00
readbuffer - > write ( buf , ret ) ;
active = true ;
have_received = true ;
}
}
if ( connected ) {
// handle outgoing packets
2024-01-09 16:36:47 -04:00
uint32_t available ;
{
WITH_SEMAPHORE ( sem ) ;
available = writebuffer - > available ( ) ;
available = MIN ( 300U , available ) ;
2023-11-28 17:20:11 -04:00
# if HAL_GCS_ENABLED
2024-01-09 16:36:47 -04:00
if ( packetise ) {
available = mavlink_packetise ( * writebuffer , available ) ;
}
2023-11-28 17:20:11 -04:00
# endif
2024-01-09 16:36:47 -04:00
if ( available = = 0 ) {
return active ;
}
}
uint8_t buf [ available ] ;
uint32_t n ;
{
WITH_SEMAPHORE ( sem ) ;
n = writebuffer - > peekbytes ( buf , available ) ;
}
if ( n > 0 ) {
const auto ret = sock - > send ( buf , n ) ;
if ( ret > 0 ) {
WITH_SEMAPHORE ( sem ) ;
writebuffer - > advance ( ret ) ;
active = true ;
2023-11-25 20:46:07 -04:00
}
}
} else {
if ( type = = NetworkPortType : : UDP_SERVER & & have_received ) {
// connect the socket to the last receive address if we have one
char buf [ 16 ] ;
uint16_t last_port ;
const char * last_addr = sock - > last_recv_address ( buf , sizeof ( buf ) , last_port ) ;
if ( last_addr ! = nullptr & & port ! = 0 ) {
connected = sock - > connect ( last_addr , last_port ) ;
2023-11-15 23:46:08 -04:00
}
}
}
2023-11-25 20:46:07 -04:00
return active ;
2023-11-15 23:46:08 -04:00
}
/*
available space in outgoing buffer
*/
uint32_t AP_Networking : : Port : : txspace ( void )
{
2023-11-18 17:46:46 -04:00
WITH_SEMAPHORE ( sem ) ;
2023-11-15 23:46:08 -04:00
return writebuffer - > space ( ) ;
}
void AP_Networking : : Port : : _begin ( uint32_t b , uint16_t rxS , uint16_t txS )
{
2023-11-18 17:46:46 -04:00
rxS = MAX ( rxS , AP_NETWORKING_PORT_MIN_RXSIZE ) ;
txS = MAX ( txS , AP_NETWORKING_PORT_MIN_TXSIZE ) ;
init_buffers ( rxS , txS ) ;
2023-11-15 23:46:08 -04:00
}
size_t AP_Networking : : Port : : _write ( const uint8_t * buffer , size_t size )
{
2023-11-18 17:46:46 -04:00
WITH_SEMAPHORE ( sem ) ;
2023-11-15 23:46:08 -04:00
return writebuffer - > write ( buffer , size ) ;
}
ssize_t AP_Networking : : Port : : _read ( uint8_t * buffer , uint16_t count )
{
2023-11-18 17:46:46 -04:00
WITH_SEMAPHORE ( sem ) ;
2023-11-15 23:46:08 -04:00
return readbuffer - > read ( buffer , count ) ;
}
uint32_t AP_Networking : : Port : : _available ( )
{
2023-11-18 17:46:46 -04:00
WITH_SEMAPHORE ( sem ) ;
2023-11-15 23:46:08 -04:00
return readbuffer - > available ( ) ;
}
bool AP_Networking : : Port : : _discard_input ( )
{
2023-11-18 17:46:46 -04:00
WITH_SEMAPHORE ( sem ) ;
2023-11-15 23:46:08 -04:00
readbuffer - > clear ( ) ;
return true ;
}
/*
initialise read / write buffers
*/
2023-11-17 15:20:39 -04:00
bool AP_Networking : : Port : : init_buffers ( const uint32_t size_rx , const uint32_t size_tx )
2023-11-15 23:46:08 -04:00
{
2023-11-18 17:46:46 -04:00
if ( size_tx = = last_size_tx & &
size_rx = = last_size_rx ) {
return true ;
}
WITH_SEMAPHORE ( sem ) ;
if ( readbuffer = = nullptr ) {
readbuffer = new ByteBuffer ( size_rx ) ;
} else {
readbuffer - > set_size_best ( size_rx ) ;
}
if ( writebuffer = = nullptr ) {
writebuffer = new ByteBuffer ( size_tx ) ;
} else {
writebuffer - > set_size_best ( size_tx ) ;
}
last_size_rx = size_rx ;
last_size_tx = size_tx ;
2023-11-15 23:46:08 -04:00
return readbuffer ! = nullptr & & writebuffer ! = nullptr ;
}
2023-11-25 20:46:07 -04:00
/*
return flow control state
*/
enum AP_HAL : : UARTDriver : : flow_control AP_Networking : : Port : : get_flow_control ( void )
{
const NetworkPortType ptype = ( NetworkPortType ) type ;
switch ( ptype ) {
case NetworkPortType : : TCP_CLIENT :
case NetworkPortType : : TCP_SERVER :
return AP_HAL : : UARTDriver : : FLOW_CONTROL_ENABLE ;
case NetworkPortType : : UDP_CLIENT :
case NetworkPortType : : UDP_SERVER :
case NetworkPortType : : NONE :
break ;
}
return AP_HAL : : UARTDriver : : FLOW_CONTROL_DISABLE ;
}
2024-01-24 00:43:53 -04:00
# endif // AP_NETWORKING_REGISTER_PORT_ENABLED