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>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "net/ipv4/addr.h"
|
|
|
|
#include "net/ipv6/addr.h"
|
|
|
|
#include "net/ipv6/hdr.h"
|
|
|
|
#include "net/sock/ip.h"
|
|
|
|
#include "timex.h"
|
|
|
|
|
|
|
|
#include "lwip/api.h"
|
|
|
|
#include "lwip/ip4.h"
|
|
|
|
#include "lwip/ip6.h"
|
|
|
|
#include "lwip/opt.h"
|
|
|
|
#include "lwip/sys.h"
|
|
|
|
#include "lwip/sock_internal.h"
|
|
|
|
|
|
|
|
int sock_ip_create(sock_ip_t *sock, const sock_ip_ep_t *local,
|
|
|
|
const sock_ip_ep_t *remote, uint8_t proto, uint16_t flags)
|
|
|
|
{
|
|
|
|
assert(sock != NULL);
|
|
|
|
|
|
|
|
int res;
|
|
|
|
struct netconn *tmp = NULL;
|
|
|
|
|
|
|
|
/* we pay attention in lwip_sock_create that _sock_tl_ep::port is not
|
|
|
|
* touched for RAW */
|
|
|
|
if ((res = lwip_sock_create(&tmp, (struct _sock_tl_ep *)local,
|
|
|
|
(struct _sock_tl_ep *)remote, proto, flags,
|
|
|
|
NETCONN_RAW)) == 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_ip_close(sock_ip_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_ip_get_local(sock_ip_t *sock, sock_ip_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_ip_get_remote(sock_ip_t *sock, sock_ip_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;
|
|
|
|
}
|
|
|
|
|
|
|
|
#if LWIP_IPV4
|
|
|
|
static uint16_t _ip4_addr_to_netif(const ip4_addr_p_t *addr)
|
|
|
|
{
|
|
|
|
assert(addr != NULL);
|
|
|
|
|
|
|
|
if (!ip4_addr_isany(addr)) {
|
|
|
|
for (struct netif *netif = netif_list; netif != NULL; netif = netif->next) {
|
|
|
|
if (netif_ip4_addr(netif)->addr == addr->addr) {
|
|
|
|
return (int)netif->num + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return SOCK_ADDR_ANY_NETIF;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if LWIP_IPV6
|
|
|
|
static uint16_t _ip6_addr_to_netif(const ip6_addr_p_t *_addr)
|
|
|
|
{
|
|
|
|
ip6_addr_t addr;
|
|
|
|
|
|
|
|
assert(_addr != NULL);
|
2018-09-04 15:55:03 +02:00
|
|
|
ip6_addr_copy_from_packed(addr, *_addr);
|
2016-09-02 19:03:35 +02:00
|
|
|
if (!ip6_addr_isany_val(addr)) {
|
|
|
|
for (struct netif *netif = netif_list; netif != NULL; netif = netif->next) {
|
|
|
|
if (netif_get_ip6_addr_match(netif, &addr) >= 0) {
|
|
|
|
return (int)netif->num + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return SOCK_ADDR_ANY_NETIF;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-03-24 17:47:50 +01:00
|
|
|
static int _parse_iphdr(struct netbuf *buf, void **data, void **ctx,
|
2016-09-02 19:03:35 +02:00
|
|
|
sock_ip_ep_t *remote)
|
|
|
|
{
|
2020-03-24 17:47:50 +01:00
|
|
|
uint8_t *data_ptr = buf->ptr->payload;
|
|
|
|
size_t data_len = buf->ptr->len;
|
2016-09-02 19:03:35 +02:00
|
|
|
|
|
|
|
switch (data_ptr[0] >> 4) {
|
|
|
|
#if LWIP_IPV4
|
|
|
|
case 4:
|
|
|
|
if (remote != NULL) {
|
|
|
|
struct ip_hdr *iphdr = (struct ip_hdr *)data_ptr;
|
|
|
|
|
|
|
|
assert(buf->p->len > sizeof(struct ip_hdr));
|
|
|
|
remote->family = AF_INET;
|
|
|
|
memcpy(&remote->addr, &iphdr->src, sizeof(ip4_addr_t));
|
|
|
|
remote->netif = _ip4_addr_to_netif(&iphdr->dest);
|
|
|
|
}
|
|
|
|
data_ptr += sizeof(struct ip_hdr);
|
|
|
|
data_len -= sizeof(struct ip_hdr);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#if LWIP_IPV6
|
|
|
|
case 6:
|
|
|
|
if (remote != NULL) {
|
|
|
|
struct ip6_hdr *iphdr = (struct ip6_hdr *)data_ptr;
|
|
|
|
|
|
|
|
assert(buf->p->len > sizeof(struct ip6_hdr));
|
|
|
|
remote->family = AF_INET6;
|
|
|
|
memcpy(&remote->addr, &iphdr->src, sizeof(ip6_addr_t));
|
|
|
|
remote->netif = _ip6_addr_to_netif(&iphdr->dest);
|
|
|
|
}
|
|
|
|
data_ptr += sizeof(struct ip6_hdr);
|
|
|
|
data_len -= sizeof(struct ip6_hdr);
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
2020-03-24 17:47:50 +01:00
|
|
|
netbuf_delete(buf);
|
2016-09-02 19:03:35 +02:00
|
|
|
return -EPROTO;
|
|
|
|
}
|
2020-03-24 17:47:50 +01:00
|
|
|
*data = data_ptr;
|
|
|
|
*ctx = buf;
|
2016-09-02 19:03:35 +02:00
|
|
|
return (ssize_t)data_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t sock_ip_recv(sock_ip_t *sock, void *data, size_t max_len,
|
|
|
|
uint32_t timeout, sock_ip_ep_t *remote)
|
2020-03-24 17:47:50 +01:00
|
|
|
{
|
|
|
|
void *pkt = NULL;
|
|
|
|
struct netbuf *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_ip_recv_buf(sock, &pkt, (void **)&ctx, timeout,
|
|
|
|
remote)) > 0) {
|
|
|
|
if (ctx->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_ip_recv_buf(sock_ip_t *sock, void **data, void **ctx,
|
|
|
|
uint32_t timeout, sock_ip_ep_t *remote)
|
2016-09-02 19:03:35 +02:00
|
|
|
{
|
|
|
|
struct netbuf *buf;
|
|
|
|
int res;
|
|
|
|
|
2020-03-24 17:47:50 +01:00
|
|
|
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;
|
|
|
|
}
|
2020-03-24 17:47:50 +01:00
|
|
|
res = _parse_iphdr(buf, data, ctx, remote);
|
2016-09-02 19:03:35 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t sock_ip_send(sock_ip_t *sock, const void *data, size_t len,
|
|
|
|
uint8_t proto, const sock_ip_ep_t *remote)
|
|
|
|
{
|
|
|
|
assert((sock != NULL) || (remote != NULL));
|
|
|
|
assert((len == 0) || (data != NULL)); /* (len != 0) => (data != NULL) */
|
2020-02-20 12:09:20 +01:00
|
|
|
return lwip_sock_send(sock ? sock->base.conn : NULL, data, len, proto,
|
2016-09-02 19:03:35 +02:00
|
|
|
(struct _sock_tl_ep *)remote, NETCONN_RAW);
|
|
|
|
}
|
|
|
|
|
2020-02-20 12:51:19 +01:00
|
|
|
#ifdef SOCK_HAS_ASYNC
|
2020-03-11 12:17:30 +01:00
|
|
|
void sock_ip_set_cb(sock_ip_t *sock, sock_ip_cb_t cb, void *arg)
|
2020-02-20 12:51:19 +01:00
|
|
|
{
|
2020-03-11 12:23:24 +01:00
|
|
|
sock->base.async_cb_arg = arg;
|
2020-02-20 12:51:19 +01:00
|
|
|
sock->base.async_cb.ip = cb;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SOCK_HAS_ASYNC_CTX
|
|
|
|
sock_async_ctx_t *sock_ip_get_async_ctx(sock_ip_t *sock)
|
|
|
|
{
|
|
|
|
return &sock->base.async_ctx;
|
|
|
|
}
|
|
|
|
#endif /* SOCK_HAS_ASYNC_CTX */
|
|
|
|
#endif /* SOCK_HAS_ASYNC */
|
|
|
|
|
2016-09-02 19:03:35 +02:00
|
|
|
/** @} */
|