1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-15 20:32:43 +01:00
RIOT/pkg/lwip/contrib/sock/udp/lwip_sock_udp.c

196 lines
5.6 KiB
C
Raw Normal View History

2016-09-02 19:03:35 +02:00
/*
* Copyright (C) 2016 Freie Universität Berlin
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @{
*
* @file
* @author Martine Lenders <m.lenders@fu-berlin.de>
*/
2020-10-21 15:58:21 +02:00
#include <assert.h>
2016-09-02 19:03:35 +02:00
#include <errno.h>
#include "net/ipv4/addr.h"
#include "net/ipv6/addr.h"
#include "net/sock/udp.h"
#include "timex.h"
#include "lwip/api.h"
#include "lwip/opt.h"
#include "lwip/sock_internal.h"
#include "lwip/sys.h"
#include "lwip/udp.h"
2016-09-02 19:03:35 +02:00
int sock_udp_create(sock_udp_t *sock, const sock_udp_ep_t *local,
const sock_udp_ep_t *remote, uint16_t flags)
{
assert(sock != NULL);
assert(remote == NULL || remote->port != 0);
int res;
struct netconn *tmp = NULL;
if ((res = lwip_sock_create(&tmp, (struct _sock_tl_ep *)local,
(struct _sock_tl_ep *)remote, 0, flags,
NETCONN_UDP)) == 0) {
2020-02-20 12:09:20 +01:00
sock->base.conn = tmp;
2020-02-20 12:51:19 +01:00
#if IS_ACTIVE(SOCK_HAS_ASYNC)
netconn_set_callback_arg(sock->base.conn, &sock->base);
#endif
2016-09-02 19:03:35 +02:00
}
return res;
}
void sock_udp_close(sock_udp_t *sock)
{
assert(sock != NULL);
2020-02-20 12:09:20 +01:00
if (sock->base.conn != NULL) {
netconn_delete(sock->base.conn);
sock->base.conn = NULL;
2016-09-02 19:03:35 +02:00
}
}
int sock_udp_get_local(sock_udp_t *sock, sock_udp_ep_t *ep)
{
assert(sock != NULL);
2020-02-20 12:09:20 +01:00
return (lwip_sock_get_addr(sock->base.conn, (struct _sock_tl_ep *)ep,
2016-09-02 19:03:35 +02:00
1)) ? -EADDRNOTAVAIL : 0;
}
int sock_udp_get_remote(sock_udp_t *sock, sock_udp_ep_t *ep)
{
assert(sock != NULL);
2020-02-20 12:09:20 +01:00
return (lwip_sock_get_addr(sock->base.conn, (struct _sock_tl_ep *)ep,
2016-09-02 19:03:35 +02:00
0)) ? -ENOTCONN : 0;
}
ssize_t sock_udp_recv_aux(sock_udp_t *sock, void *data, size_t max_len,
uint32_t timeout, sock_udp_ep_t *remote,
sock_udp_aux_rx_t *aux)
2016-09-02 19:03:35 +02:00
{
void *pkt = NULL;
void *ctx = NULL;
uint8_t *ptr = data;
ssize_t res, ret = 0;
bool nobufs = false;
assert((sock != NULL) && (data != NULL) && (max_len > 0));
while ((res = sock_udp_recv_buf_aux(sock, &pkt, &ctx, timeout,
remote, aux)) > 0) {
struct netbuf *buf = ctx;
if (buf->p->tot_len > (ssize_t)max_len) {
nobufs = true;
/* progress context to last element */
while (netbuf_next(ctx) == 0) {}
continue;
}
memcpy(ptr, pkt, res);
ptr += res;
ret += res;
}
return (nobufs) ? -ENOBUFS : ((res < 0) ? res : ret);
}
ssize_t sock_udp_recv_buf_aux(sock_udp_t *sock, void **data, void **ctx,
uint32_t timeout, sock_udp_ep_t *remote,
sock_udp_aux_rx_t *aux)
{
(void)aux;
2016-09-02 19:03:35 +02:00
struct netbuf *buf;
int res;
assert((sock != NULL) && (data != NULL) && (ctx != NULL));
buf = *ctx;
if (buf != NULL) {
if (netbuf_next(buf) == -1) {
*data = NULL;
netbuf_delete(buf);
*ctx = NULL;
return 0;
}
else {
*data = buf->ptr->payload;
return buf->ptr->len;
}
}
2020-02-20 12:09:20 +01:00
if ((res = lwip_sock_recv(sock->base.conn, timeout, &buf)) < 0) {
2016-09-02 19:03:35 +02:00
return res;
}
if ((remote != NULL) ||
((aux != NULL) && IS_USED(MODULE_SOCK_AUX_LOCAL)
&& IS_ACTIVE(LWIP_NETBUF_RECVINFO))) {
2016-09-02 19:03:35 +02:00
/* convert remote */
size_t addr_len = sizeof(ipv4_addr_t);
int family = AF_INET;
if (NETCONNTYPE_ISIPV6(sock->base.conn->type)) {
2016-09-02 19:03:35 +02:00
addr_len = sizeof(ipv6_addr_t);
family = AF_INET6;
}
else if (!IS_ACTIVE(LWIP_IPV4)) {
2016-09-02 19:03:35 +02:00
netbuf_delete(buf);
return -EPROTO;
}
if (remote != NULL) {
remote->family = family;
2016-09-02 19:03:35 +02:00
#if LWIP_NETBUF_RECVINFO
remote->netif = lwip_sock_bind_addr_to_netif(&buf->toaddr);
2016-09-02 19:03:35 +02:00
#else
remote->netif = SOCK_ADDR_ANY_NETIF;
2016-09-02 19:03:35 +02:00
#endif
/* copy address */
memcpy(&remote->addr, &buf->addr, addr_len);
remote->port = buf->port;
}
#if IS_USED(MODULE_SOCK_AUX_LOCAL)
static_assert(IS_ACTIVE(LWIP_NETBUF_RECVINFO),
"sock_aux_local depends on LWIP_NETBUF_RECVINFO");
if ((aux != NULL) && (aux->flags & SOCK_AUX_GET_LOCAL)) {
aux->flags &= ~(SOCK_AUX_GET_LOCAL);
aux->local.family = family;
memcpy(&aux->local.addr, &buf->toaddr, addr_len);
aux->local.port = sock->base.conn->pcb.udp->local_port;
}
#endif /* MODULE_SOCK_AUX_LOCAL */
2016-09-02 19:03:35 +02:00
}
*data = buf->ptr->payload;
*ctx = buf;
return (ssize_t)buf->ptr->len;
2016-09-02 19:03:35 +02:00
}
ssize_t sock_udp_send_aux(sock_udp_t *sock, const void *data, size_t len,
const sock_udp_ep_t *remote, sock_udp_aux_tx_t *aux)
2016-09-02 19:03:35 +02:00
{
(void)aux;
2016-09-02 19:03:35 +02:00
assert((sock != NULL) || (remote != NULL));
assert((len == 0) || (data != NULL)); /* (len != 0) => (data != NULL) */
if ((remote != NULL) && (remote->port == 0)) {
return -EINVAL;
}
2020-02-20 12:09:20 +01:00
return lwip_sock_send((sock) ? sock->base.conn : NULL, data, len, 0,
(struct _sock_tl_ep *)remote, NETCONN_UDP);
2016-09-02 19:03:35 +02:00
}
2020-02-20 12:51:19 +01:00
#ifdef SOCK_HAS_ASYNC
void sock_udp_set_cb(sock_udp_t *sock, sock_udp_cb_t cb, void *arg)
2020-02-20 12:51:19 +01:00
{
sock->base.async_cb_arg = arg;
2020-02-20 12:51:19 +01:00
sock->base.async_cb.udp = cb;
}
#ifdef SOCK_HAS_ASYNC_CTX
sock_async_ctx_t *sock_udp_get_async_ctx(sock_udp_t *sock)
{
return &sock->base.async_ctx;
}
#endif /* SOCK_HAS_ASYNC_CTX */
#endif /* SOCK_HAS_ASYNC */
2016-09-02 19:03:35 +02:00
/** @} */