1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00

Merge pull request #16493 from brummer-simon/gnrc_tcp-align_with_sock_tcp

gnrc_tcp: align with sock tcp
This commit is contained in:
benpicco 2021-07-13 22:00:17 +02:00 committed by GitHub
commit 602384eb86
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 395 additions and 8 deletions

View File

@ -101,6 +101,14 @@ int gnrc_tcp_init(void);
*/ */
void gnrc_tcp_tcb_init(gnrc_tcp_tcb_t *tcb); void gnrc_tcp_tcb_init(gnrc_tcp_tcb_t *tcb);
/**
* @brief Initialize Transmission Control Block (TCB) queue
* @pre @p queue must not be NULL.
*
* @param[in,out] queue TCB queue to initialize.
*/
void gnrc_tcp_tcb_queue_init(gnrc_tcp_tcb_queue_t *queue);
/** /**
* @brief Opens a connection. * @brief Opens a connection.
* *
@ -168,6 +176,7 @@ int gnrc_tcp_listen(gnrc_tcp_tcb_queue_t *queue, gnrc_tcp_tcb_t *tcbs, size_t tc
* *
* @return 0 on success. * @return 0 on success.
* @return -ENOMEM if all connection in @p queue were already accepted. * @return -ENOMEM if all connection in @p queue were already accepted.
* @return -EINVAL if listen was never called on queue.
* @return -EAGAIN if @p user_timeout_duration_ms was 0 and no connection is ready to accept. * @return -EAGAIN if @p user_timeout_duration_ms was 0 and no connection is ready to accept.
* @return -ETIMEDOUT if @p user_timeout_duration_ms was not 0 and no connection * @return -ETIMEDOUT if @p user_timeout_duration_ms was not 0 and no connection
* could be established. * could be established.
@ -263,6 +272,48 @@ void gnrc_tcp_abort(gnrc_tcp_tcb_t *tcb);
*/ */
void gnrc_tcp_stop_listen(gnrc_tcp_tcb_queue_t *queue); void gnrc_tcp_stop_listen(gnrc_tcp_tcb_queue_t *queue);
/**
* @brief Get the local end point of a connected TCB
*
* @pre tcb must not be NULL
* @pre ep must not be NULL
*
* @param[in] tcb TCB holding the connection information.
* @param[out] ep The local end point.
*
* @return 0 on success.
* @return -EADDRNOTAVAIL, when @p tcb in not in a connected state.
*/
int gnrc_tcp_get_local(gnrc_tcp_tcb_t *tcb, gnrc_tcp_ep_t *ep);
/**
* @brief Get the remote end point of a connected TCB
*
* @pre tcb must not be NULL
* @pre ep must not be NULL
*
* @param[in] tcb TCB holding the connection information.
* @param[out] ep The remote end point.
*
* @return 0 on success.
* @return -ENOTCONN, when @p tcb in not in a connected state.
*/
int gnrc_tcp_get_remote(gnrc_tcp_tcb_t *tcb, gnrc_tcp_ep_t *ep);
/**
* @brief Gets the local end point of a TCB queue
*
* @pre queue must not be NULL
* @pre ep must not be NULL
*
* @param[in] queue TCB queue to stop listening
* @param[out] ep The local end point.
*
* @return 0 on success.
* @return -EADDRNOTAVAIL, when @p queue has no local end point.
*/
int gnrc_tcp_queue_get_local(gnrc_tcp_tcb_queue_t *queue, gnrc_tcp_ep_t *ep);
/** /**
* @brief Calculate and set checksum in TCP header. * @brief Calculate and set checksum in TCP header.
* *

View File

@ -310,6 +310,8 @@ int gnrc_tcp_init(void)
void gnrc_tcp_tcb_init(gnrc_tcp_tcb_t *tcb) void gnrc_tcp_tcb_init(gnrc_tcp_tcb_t *tcb)
{ {
TCP_DEBUG_ENTER; TCP_DEBUG_ENTER;
assert(tcb != NULL);
memset(tcb, 0, sizeof(gnrc_tcp_tcb_t)); memset(tcb, 0, sizeof(gnrc_tcp_tcb_t));
#ifdef MODULE_GNRC_IPV6 #ifdef MODULE_GNRC_IPV6
tcb->address_family = AF_INET6; tcb->address_family = AF_INET6;
@ -324,6 +326,16 @@ void gnrc_tcp_tcb_init(gnrc_tcp_tcb_t *tcb)
TCP_DEBUG_LEAVE; TCP_DEBUG_LEAVE;
} }
void gnrc_tcp_tcb_queue_init(gnrc_tcp_tcb_queue_t *queue)
{
TCP_DEBUG_ENTER;
assert(queue != NULL);
mutex_init(&queue->lock);
queue->tcbs = NULL;
queue->tcbs_len = 0;
TCP_DEBUG_LEAVE;
}
int gnrc_tcp_open(gnrc_tcp_tcb_t *tcb, const gnrc_tcp_ep_t *remote, uint16_t local_port) int gnrc_tcp_open(gnrc_tcp_tcb_t *tcb, const gnrc_tcp_ep_t *remote, uint16_t local_port)
{ {
@ -561,14 +573,19 @@ int gnrc_tcp_accept(gnrc_tcp_tcb_queue_t *queue, gnrc_tcp_tcb_t **tcb,
++avail_tcbs; ++avail_tcbs;
} }
/* Return if a connection was found, accept was called as non-blocking or all /* Return if a connection was found, queue is not listening, accept was called as non-blocking
* TCBs were already accepted. * or all TCBs were already accepted.
*/ */
if ((*tcb) || (user_timeout_duration_ms == 0) || (avail_tcbs == 0)) { if ((*tcb) || (queue->tcbs == NULL) || (user_timeout_duration_ms == 0) ||
(avail_tcbs == 0)) {
if (*tcb) { if (*tcb) {
TCP_DEBUG_INFO("Accepting connection."); TCP_DEBUG_INFO("Accepting connection.");
ret = 0; ret = 0;
} }
else if (queue->tcbs == NULL) {
TCP_DEBUG_ERROR("-EINVAL: Queue is not listening.");
ret = -EINVAL;
}
else if (avail_tcbs == 0) { else if (avail_tcbs == 0) {
TCP_DEBUG_ERROR("-ENOMEM: All TCBs are currently accepted."); TCP_DEBUG_ERROR("-ENOMEM: All TCBs are currently accepted.");
ret = -ENOMEM; ret = -ENOMEM;
@ -933,6 +950,113 @@ void gnrc_tcp_stop_listen(gnrc_tcp_tcb_queue_t *queue)
TCP_DEBUG_LEAVE; TCP_DEBUG_LEAVE;
} }
int gnrc_tcp_get_local(gnrc_tcp_tcb_t *tcb, gnrc_tcp_ep_t *ep)
{
TCP_DEBUG_ENTER;
assert(tcb != NULL);
assert(ep != NULL);
int ret = 0;
/* Lock the TCB for this function call */
mutex_lock(&(tcb->function_lock));
_gnrc_tcp_fsm_state_t state =_gnrc_tcp_fsm_get_state(tcb);
/* Check if connection is established */
if ((state == FSM_STATE_ESTABLISHED) || (state == FSM_STATE_CLOSE_WAIT)) {
/* Construct endpoint from connection parameters */
ep->family = tcb->address_family;
ep->port = tcb->local_port;
ep->netif = tcb->ll_iface;
#ifdef MODULE_GNRC_IPV6
if (ep->family == AF_INET6) {
memcpy(ep->addr.ipv6, tcb->local_addr, sizeof(ep->addr.ipv6));
}
#endif
} else {
TCP_DEBUG_ERROR("-EADDRNOTAVAIL: TCB is not connected.");
ret = -EADDRNOTAVAIL;
}
mutex_unlock(&(tcb->function_lock));
TCP_DEBUG_LEAVE;
return ret;
}
int gnrc_tcp_get_remote(gnrc_tcp_tcb_t *tcb, gnrc_tcp_ep_t *ep)
{
TCP_DEBUG_ENTER;
assert(tcb != NULL);
assert(ep != NULL);
int ret = 0;
/* Lock the TCB for this function call */
mutex_lock(&(tcb->function_lock));
_gnrc_tcp_fsm_state_t state =_gnrc_tcp_fsm_get_state(tcb);
/* Check if connection is established */
if ((state == FSM_STATE_ESTABLISHED) || (state == FSM_STATE_CLOSE_WAIT)) {
/* Construct endpoint from connection parameters */
ep->family = tcb->address_family;
ep->port = tcb->peer_port;
ep->netif = 0;
#ifdef MODULE_GNRC_IPV6
if (ep->family == AF_INET6) {
memcpy(ep->addr.ipv6, tcb->peer_addr, sizeof(ep->addr.ipv6));
}
#endif
} else {
TCP_DEBUG_ERROR("-ENOTCONN: TCB is not connected.");
ret = -ENOTCONN;
}
mutex_unlock(&(tcb->function_lock));
TCP_DEBUG_LEAVE;
return ret;
}
int gnrc_tcp_queue_get_local(gnrc_tcp_tcb_queue_t *queue, gnrc_tcp_ep_t *ep)
{
TCP_DEBUG_ENTER;
assert(queue != NULL);
assert(ep != NULL);
int ret = 0;
/* Lock the TCB queue for this function call */
mutex_lock(&(queue->lock));
/* Check if queue has associated TCBs */
if (queue->tcbs) {
/* There are listening TCBs: Construct ep from first TCB. */
gnrc_tcp_tcb_t *tcb = queue->tcbs;
mutex_lock(&(tcb->function_lock));
/* Construct endpoint from tcbs connection parameters */
ep->family = tcb->address_family;
ep->port = tcb->local_port;
ep->netif = 0;
#ifdef MODULE_GNRC_IPV6
if (ep->family == AF_INET6) {
if (tcb->status & STATUS_ALLOW_ANY_ADDR) {
ipv6_addr_set_unspecified((ipv6_addr_t *) ep->addr.ipv6);
} else {
memcpy(ep->addr.ipv6, tcb->local_addr, sizeof(ep->addr.ipv6));
}
}
#endif
mutex_unlock(&(tcb->function_lock));
} else {
TCP_DEBUG_ERROR("-EADDRNOTAVAIL: queue was never listening.");
ret = -EADDRNOTAVAIL;
}
mutex_unlock(&(queue->lock));
TCP_DEBUG_LEAVE;
return ret;
}
int gnrc_tcp_calc_csum(const gnrc_pktsnip_t *hdr, const gnrc_pktsnip_t *pseudo_hdr) int gnrc_tcp_calc_csum(const gnrc_pktsnip_t *hdr, const gnrc_pktsnip_t *pseudo_hdr)
{ {
TCP_DEBUG_ENTER; TCP_DEBUG_ENTER;

View File

@ -217,6 +217,10 @@ int gnrc_tcp_accept_cmd(int argc, char **argv)
int timeout = atol(argv[1]); int timeout = atol(argv[1]);
int err = gnrc_tcp_accept(&queue, &tmp, timeout); int err = gnrc_tcp_accept(&queue, &tmp, timeout);
switch (err) { switch (err) {
case -EINVAL:
printf("%s: returns -EINVAL\n", argv[0]);
break;
case -EAGAIN: case -EAGAIN:
printf("%s: returns -EAGAIN\n", argv[0]); printf("%s: returns -EAGAIN\n", argv[0]);
break; break;
@ -341,6 +345,78 @@ int gnrc_tcp_stop_listen_cmd(int argc, char **argv)
return 0; return 0;
} }
int gnrc_tcp_get_local_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_ep_t ep;
int err = gnrc_tcp_get_local(tcb, &ep);
switch (err) {
case 0:
printf("%s: returns 0\n", argv[0]);
printf("Endpoint: addr.ipv6=");
ipv6_addr_print((ipv6_addr_t *) ep.addr.ipv6);
printf(" netif=%u port=%u\n", ep.netif, ep.port);
break;
case -EADDRNOTAVAIL:
printf("%s: returns -EADDRNOTAVAIL\n", argv[0]);
break;
default:
printf("%s: returns %d\n", argv[0], err);
}
return 0;
}
int gnrc_tcp_get_remote_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_ep_t ep;
int err = gnrc_tcp_get_remote(tcb, &ep);
switch (err) {
case 0:
printf("%s: returns 0\n", argv[0]);
printf("Endpoint: addr.ipv6=");
ipv6_addr_print((ipv6_addr_t *) ep.addr.ipv6);
printf(" netif=%u port=%u\n", ep.netif, ep.port);
break;
case -ENOTCONN:
printf("%s: returns -ENOTCONN\n", argv[0]);
break;
default:
printf("%s: returns %d\n", argv[0], err);
}
return 0;
}
int gnrc_tcp_queue_get_local_cmd(int argc, char **argv)
{
dump_args(argc, argv);
gnrc_tcp_ep_t ep;
int err = gnrc_tcp_queue_get_local(&queue, &ep);
switch (err) {
case 0:
printf("%s: returns 0\n", argv[0]);
printf("Endpoint: addr.ipv6=");
ipv6_addr_print((ipv6_addr_t *) ep.addr.ipv6);
printf(" netif=%u port=%u\n", ep.netif, ep.port);
break;
case -EADDRNOTAVAIL:
printf("%s: returns -EADDRNOTAVAIL\n", argv[0]);
break;
default:
printf("%s: returns %d\n", argv[0], err);
}
return 0;
}
/* Exporting GNRC TCP Api to for shell usage */ /* Exporting GNRC TCP Api to for shell usage */
static const shell_command_t shell_commands[] = { static const shell_command_t shell_commands[] = {
{ "gnrc_tcp_ep_from_str", "Build endpoint from string", { "gnrc_tcp_ep_from_str", "Build endpoint from string",
@ -363,6 +439,12 @@ static const shell_command_t shell_commands[] = {
gnrc_tcp_abort_cmd }, gnrc_tcp_abort_cmd },
{ "gnrc_tcp_stop_listen", "gnrc_tcp: stop listening", { "gnrc_tcp_stop_listen", "gnrc_tcp: stop listening",
gnrc_tcp_stop_listen_cmd }, gnrc_tcp_stop_listen_cmd },
{ "gnrc_tcp_get_local", "gnrc_tcp: get local",
gnrc_tcp_get_local_cmd },
{ "gnrc_tcp_get_remote", "gnrc_tcp: get remote",
gnrc_tcp_get_remote_cmd },
{ "gnrc_tcp_queue_get_local", "gnrc_tcp: get queue local",
gnrc_tcp_queue_get_local_cmd },
{ "buffer_init", "init internal buffer", { "buffer_init", "init internal buffer",
buffer_init_cmd }, buffer_init_cmd },
{ "buffer_get_max_size", "get max size of internal buffer", { "buffer_get_max_size", "get max size of internal buffer",

View File

@ -183,7 +183,7 @@ def test_gnrc_tcp_garbage_packets_ack_instead_of_sym(child):
riot_srv.close() riot_srv.close()
@Runner(timeout=5, skip=False) @Runner(timeout=5)
def test_gnrc_tcp_garbage_packets_option_parsing(child): def test_gnrc_tcp_garbage_packets_option_parsing(child):
""" This test verfies that malformed option don't break TCP """ This test verfies that malformed option don't break TCP
doesn't break GNRC_TCP. See: https://github.com/RIOT-OS/RIOT/issues/12086 doesn't break GNRC_TCP. See: https://github.com/RIOT-OS/RIOT/issues/12086
@ -394,6 +394,118 @@ def test_gnrc_tcp_accept_returns_ENOMEM(child):
child.expect_exact('gnrc_tcp_accept: returns -ENOMEM') child.expect_exact('gnrc_tcp_accept: returns -ENOMEM')
@Runner(timeout=0.5)
def test_gnrc_tcp_accept_returns_EINVAL(child):
""" gnrc_tcp_accept must return with -EINVAL
if listen was not called before.
"""
child.sendline('gnrc_tcp_accept 0')
child.expect_exact('gnrc_tcp_accept: returns -EINVAL')
@Runner(timeout=5)
def test_gnrc_tcp_get_local_returns_0(child):
""" This test verifies that get_local returns 0 in a connected state
and the used endpoint contains the expected connection parameters
"""
local_port = 30423
# Setup Host as server
with HostTcpServer(generate_port_number()) as host_srv:
# Setup Riot as client
with RiotTcpClient(child, host_srv, local_port) as riot_cli:
# Accept connection
host_srv.accept()
# Get and verify local endpoint
riot_cli.get_local()
child.expect_exact('Endpoint: addr.ipv6={} netif={} port={}'.format(
riot_cli.address, riot_cli.interface, riot_cli.port)
)
# Close connection
host_srv.close()
@Runner(timeout=1)
def test_gnrc_tcp_get_local_returns_EADDRNOTAVAIL(child):
""" This Test verifies that get_local returns -EADDRNOTAVAIL
then not connected
"""
child.sendline('gnrc_tcp_get_local')
child.expect_exact('gnrc_tcp_get_local: returns -EADDRNOTAVAIL')
@Runner(timeout=5)
def test_gnrc_tcp_get_remote_returns_0(child):
""" This test verifies that get_remote returns 0 in a connected state
and the used endpoint contains the expected connection parameters
"""
# Setup Host as server
with HostTcpServer(generate_port_number()) as host_srv:
# Setup Riot as client
with RiotTcpClient(child, host_srv) as riot_cli:
# Accept connection
host_srv.accept()
# Get and verify local endpoint
riot_cli.get_remote()
child.expect_exact('Endpoint: addr.ipv6={} netif=0 port={}'.format(
host_srv.address, host_srv.listen_port)
)
# Close connection
host_srv.close()
@Runner(timeout=1)
def test_gnrc_tcp_get_remote_returns_ENOTCONN(child):
""" This test verifies that get_remote returns -ENOTCONN
then not connected
"""
child.sendline('gnrc_tcp_get_remote')
child.expect_exact('gnrc_tcp_get_remote: returns -ENOTCONN')
@Runner(timeout=1)
def test_gnrc_tcp_queue_get_local_returns_0(child):
""" This test verifies that queue_get_local returns 0 then put after listen.
And the endpoint content is as are as expected.
"""
# Enter listen with accept all address
listen_addr = '::'
listen_port = generate_port_number()
riot_srv = RiotTcpServer(child, listen_port)
riot_srv.listen()
riot_srv.queue_get_local()
child.expect_exact('Endpoint: addr.ipv6=:: netif=0 port={}'.format(
listen_port)
)
riot_srv.stop_listen()
# Enter listen with specified address
listen_addr = 'fe80::4c49:c7ff:fecd:34a3'
listen_port = generate_port_number()
riot_srv = RiotTcpServer(child, listen_port, listen_addr)
riot_srv.listen()
riot_srv.queue_get_local()
child.expect_exact('Endpoint: addr.ipv6={} netif=0 port={}'.format(
listen_addr, listen_port)
)
riot_srv.stop_listen()
@Runner(timeout=1)
def test_gnrc_tcp_queue_get_local_returns_EADDRNOTAVAIL(child):
""" This Test verifies that queue_get_local returns -EADDRNOTAVAIL
then not listening
"""
child.sendline('gnrc_tcp_queue_get_local')
child.expect_exact('gnrc_tcp_queue_get_local: returns -EADDRNOTAVAIL')
@Runner(timeout=10) @Runner(timeout=10)
def test_connection_listen_accept_cycle(child, iterations=10): def test_connection_listen_accept_cycle(child, iterations=10):
""" This test verifies gnrc_tcp in a typical server role by """ This test verifies gnrc_tcp in a typical server role by

View File

@ -181,6 +181,14 @@ class _RiotTcpNode:
self._verify_pktbuf_empty() self._verify_pktbuf_empty()
self.opened = False self.opened = False
def get_local(self):
self.child.sendline('gnrc_tcp_get_local')
self.child.expect_exact('gnrc_tcp_get_local: returns 0')
def get_remote(self):
self.child.sendline('gnrc_tcp_get_remote')
self.child.expect_exact('gnrc_tcp_get_remote: returns 0')
def _get_interface(self): def _get_interface(self):
self.child.sendline('ifconfig') self.child.sendline('ifconfig')
self.child.expect(r'Iface\s+(\d+)\s') self.child.expect(r'Iface\s+(\d+)\s')
@ -230,10 +238,11 @@ class _RiotTcpNode:
class RiotTcpServer(_RiotTcpNode): class RiotTcpServer(_RiotTcpNode):
def __init__(self, child, listen_port): def __init__(self, child, listen_port, listen_addr='::'):
super().__init__(child) super().__init__(child)
self.listening = False self.listening = False
self.listen_port = str(listen_port) self.listen_port = str(listen_port)
self.listen_addr = str(listen_addr)
self.tcb_init() self.tcb_init()
def __enter__(self): def __enter__(self):
@ -246,7 +255,9 @@ class RiotTcpServer(_RiotTcpNode):
self.stop_listen() self.stop_listen()
def listen(self): def listen(self):
self.child.sendline('gnrc_tcp_listen [::]:{}'.format(self.listen_port)) self.child.sendline('gnrc_tcp_listen [{}]:{}'.format(
self.listen_addr, self.listen_port)
)
self.child.expect_exact('gnrc_tcp_listen: returns 0') self.child.expect_exact('gnrc_tcp_listen: returns 0')
self.listening = True self.listening = True
@ -261,12 +272,17 @@ class RiotTcpServer(_RiotTcpNode):
self._verify_pktbuf_empty() self._verify_pktbuf_empty()
self.listening = False self.listening = False
def queue_get_local(self):
self.child.sendline('gnrc_tcp_queue_get_local')
self.child.expect_exact('gnrc_tcp_queue_get_local: returns 0')
class RiotTcpClient(_RiotTcpNode): class RiotTcpClient(_RiotTcpNode):
def __init__(self, child, target): def __init__(self, child, target, port=0):
super().__init__(child) super().__init__(child)
self.target_addr = target.address + '%' + self.interface self.target_addr = target.address + '%' + self.interface
self.target_port = str(target.listen_port) self.target_port = str(target.listen_port)
self.port = port
self.tcb_init() self.tcb_init()
def __enter__(self): def __enter__(self):
@ -279,7 +295,9 @@ class RiotTcpClient(_RiotTcpNode):
self.close() self.close()
def open(self): def open(self):
self.child.sendline('gnrc_tcp_open [{}]:{} 0'.format(self.target_addr, self.target_port)) self.child.sendline('gnrc_tcp_open [{}]:{} {}'.format(
self.target_addr, self.target_port, self.port)
)
self.child.expect_exact('gnrc_tcp_open: returns 0') self.child.expect_exact('gnrc_tcp_open: returns 0')
self.opened = True self.opened = True