2017-06-30 16:46:01 +02:00
|
|
|
/** @file
|
|
|
|
* @brief UDP packet helpers.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2017 Intel Corporation
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
2022-05-06 11:12:04 +02:00
|
|
|
#include <zephyr/logging/log.h>
|
2018-11-30 11:54:56 +01:00
|
|
|
LOG_MODULE_REGISTER(net_udp, CONFIG_NET_UDP_LOG_LEVEL);
|
2017-06-30 16:46:01 +02:00
|
|
|
|
|
|
|
#include "net_private.h"
|
|
|
|
#include "udp_internal.h"
|
2018-12-13 21:36:57 +01:00
|
|
|
#include "net_stats.h"
|
2017-06-30 16:46:01 +02:00
|
|
|
|
|
|
|
#define PKT_WAIT_TIME K_SECONDS(1)
|
|
|
|
|
2020-05-27 18:26:57 +02:00
|
|
|
int net_udp_create(struct net_pkt *pkt, uint16_t src_port, uint16_t dst_port)
|
2018-11-27 11:34:57 +01:00
|
|
|
{
|
|
|
|
NET_PKT_DATA_ACCESS_DEFINE(udp_access, struct net_udp_hdr);
|
|
|
|
struct net_udp_hdr *udp_hdr;
|
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
udp_hdr = (struct net_udp_hdr *)net_pkt_get_data(pkt, &udp_access);
|
2018-11-27 11:34:57 +01:00
|
|
|
if (!udp_hdr) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
|
|
|
udp_hdr->src_port = src_port;
|
|
|
|
udp_hdr->dst_port = dst_port;
|
2019-03-27 02:57:45 +01:00
|
|
|
udp_hdr->len = 0U;
|
|
|
|
udp_hdr->chksum = 0U;
|
2018-11-27 11:34:57 +01:00
|
|
|
|
|
|
|
return net_pkt_set_data(pkt, &udp_access);
|
|
|
|
}
|
|
|
|
|
2023-10-12 15:49:00 +02:00
|
|
|
int net_udp_finalize(struct net_pkt *pkt, bool force_chksum)
|
2018-11-27 11:34:57 +01:00
|
|
|
{
|
|
|
|
NET_PKT_DATA_ACCESS_DEFINE(udp_access, struct net_udp_hdr);
|
|
|
|
struct net_udp_hdr *udp_hdr;
|
2020-05-27 18:26:57 +02:00
|
|
|
uint16_t length = 0;
|
2018-11-27 11:34:57 +01:00
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
udp_hdr = (struct net_udp_hdr *)net_pkt_get_data(pkt, &udp_access);
|
2018-11-27 11:34:57 +01:00
|
|
|
if (!udp_hdr) {
|
|
|
|
return -ENOBUFS;
|
|
|
|
}
|
|
|
|
|
2019-09-30 11:17:03 +02:00
|
|
|
length = net_pkt_get_len(pkt) - net_pkt_ip_hdr_len(pkt) -
|
|
|
|
net_pkt_ip_opts_len(pkt);
|
|
|
|
|
2019-03-18 16:42:22 +01:00
|
|
|
udp_hdr->len = htons(length);
|
2018-11-27 11:34:57 +01:00
|
|
|
|
2023-10-12 15:49:00 +02:00
|
|
|
if (net_if_need_calc_tx_checksum(net_pkt_iface(pkt)) || force_chksum) {
|
2019-03-18 16:42:22 +01:00
|
|
|
udp_hdr->chksum = net_calc_chksum_udp(pkt);
|
2023-10-11 16:45:36 +02:00
|
|
|
net_pkt_set_chksum_done(pkt, true);
|
2019-03-18 16:42:22 +01:00
|
|
|
}
|
2018-11-27 11:34:57 +01:00
|
|
|
|
|
|
|
return net_pkt_set_data(pkt, &udp_access);
|
|
|
|
}
|
|
|
|
|
2017-06-30 16:46:01 +02:00
|
|
|
struct net_udp_hdr *net_udp_get_hdr(struct net_pkt *pkt,
|
|
|
|
struct net_udp_hdr *hdr)
|
|
|
|
{
|
2019-02-04 11:09:48 +01:00
|
|
|
NET_PKT_DATA_ACCESS_CONTIGUOUS_DEFINE(udp_access, struct net_udp_hdr);
|
|
|
|
struct net_pkt_cursor backup;
|
2017-06-30 16:46:01 +02:00
|
|
|
struct net_udp_hdr *udp_hdr;
|
2019-02-04 11:09:48 +01:00
|
|
|
bool overwrite;
|
2017-06-30 16:46:01 +02:00
|
|
|
|
2019-02-04 11:09:48 +01:00
|
|
|
udp_access.data = hdr;
|
2017-06-30 16:46:01 +02:00
|
|
|
|
2019-02-04 11:09:48 +01:00
|
|
|
overwrite = net_pkt_is_being_overwritten(pkt);
|
|
|
|
net_pkt_set_overwrite(pkt, true);
|
|
|
|
|
|
|
|
net_pkt_cursor_backup(pkt, &backup);
|
|
|
|
net_pkt_cursor_init(pkt);
|
|
|
|
|
|
|
|
if (net_pkt_skip(pkt, net_pkt_ip_hdr_len(pkt) +
|
2019-09-30 11:17:03 +02:00
|
|
|
net_pkt_ip_opts_len(pkt))) {
|
2019-02-04 11:09:48 +01:00
|
|
|
udp_hdr = NULL;
|
|
|
|
goto out;
|
2017-06-30 16:46:01 +02:00
|
|
|
}
|
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
udp_hdr = (struct net_udp_hdr *)net_pkt_get_data(pkt, &udp_access);
|
2019-02-04 11:09:48 +01:00
|
|
|
|
|
|
|
out:
|
|
|
|
net_pkt_cursor_restore(pkt, &backup);
|
|
|
|
net_pkt_set_overwrite(pkt, overwrite);
|
|
|
|
|
|
|
|
return udp_hdr;
|
2017-06-30 16:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
struct net_udp_hdr *net_udp_set_hdr(struct net_pkt *pkt,
|
|
|
|
struct net_udp_hdr *hdr)
|
|
|
|
{
|
2019-02-04 11:09:48 +01:00
|
|
|
NET_PKT_DATA_ACCESS_DEFINE(udp_access, struct net_udp_hdr);
|
|
|
|
struct net_pkt_cursor backup;
|
|
|
|
struct net_udp_hdr *udp_hdr;
|
|
|
|
bool overwrite;
|
|
|
|
|
|
|
|
overwrite = net_pkt_is_being_overwritten(pkt);
|
|
|
|
net_pkt_set_overwrite(pkt, true);
|
2017-06-30 16:46:01 +02:00
|
|
|
|
2019-02-04 11:09:48 +01:00
|
|
|
net_pkt_cursor_backup(pkt, &backup);
|
|
|
|
net_pkt_cursor_init(pkt);
|
|
|
|
|
|
|
|
if (net_pkt_skip(pkt, net_pkt_ip_hdr_len(pkt) +
|
2019-09-30 11:17:03 +02:00
|
|
|
net_pkt_ip_opts_len(pkt))) {
|
2019-02-04 11:09:48 +01:00
|
|
|
udp_hdr = NULL;
|
|
|
|
goto out;
|
2017-06-30 16:46:01 +02:00
|
|
|
}
|
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
udp_hdr = (struct net_udp_hdr *)net_pkt_get_data(pkt, &udp_access);
|
2019-02-04 11:09:48 +01:00
|
|
|
if (!udp_hdr) {
|
|
|
|
goto out;
|
2017-06-30 16:46:01 +02:00
|
|
|
}
|
|
|
|
|
2019-02-04 11:09:48 +01:00
|
|
|
memcpy(udp_hdr, hdr, sizeof(struct net_udp_hdr));
|
|
|
|
|
|
|
|
net_pkt_set_data(pkt, &udp_access);
|
|
|
|
out:
|
|
|
|
net_pkt_cursor_restore(pkt, &backup);
|
|
|
|
net_pkt_set_overwrite(pkt, overwrite);
|
|
|
|
|
|
|
|
return udp_hdr == NULL ? NULL : hdr;
|
2017-06-30 16:46:01 +02:00
|
|
|
}
|
|
|
|
|
2020-05-27 18:26:57 +02:00
|
|
|
int net_udp_register(uint8_t family,
|
2019-01-28 14:31:33 +01:00
|
|
|
const struct sockaddr *remote_addr,
|
|
|
|
const struct sockaddr *local_addr,
|
2020-05-27 18:26:57 +02:00
|
|
|
uint16_t remote_port,
|
|
|
|
uint16_t local_port,
|
2021-03-24 09:46:24 +01:00
|
|
|
struct net_context *context,
|
2019-01-28 14:31:33 +01:00
|
|
|
net_conn_cb_t cb,
|
|
|
|
void *user_data,
|
|
|
|
struct net_conn_handle **handle)
|
2017-06-30 16:46:01 +02:00
|
|
|
{
|
2019-01-28 14:31:33 +01:00
|
|
|
return net_conn_register(IPPROTO_UDP, family, remote_addr, local_addr,
|
2021-03-24 09:46:24 +01:00
|
|
|
remote_port, local_port, context, cb,
|
|
|
|
user_data, handle);
|
2017-06-30 16:46:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int net_udp_unregister(struct net_conn_handle *handle)
|
|
|
|
{
|
|
|
|
return net_conn_unregister(handle);
|
|
|
|
}
|
2018-12-13 21:36:57 +01:00
|
|
|
|
|
|
|
struct net_udp_hdr *net_udp_input(struct net_pkt *pkt,
|
|
|
|
struct net_pkt_data_access *udp_access)
|
|
|
|
{
|
|
|
|
struct net_udp_hdr *udp_hdr;
|
|
|
|
|
2019-02-20 09:28:18 +01:00
|
|
|
udp_hdr = (struct net_udp_hdr *)net_pkt_get_data(pkt, udp_access);
|
2019-02-05 09:57:49 +01:00
|
|
|
if (!udp_hdr || net_pkt_set_data(pkt, udp_access)) {
|
|
|
|
NET_DBG("DROP: corrupted header");
|
|
|
|
goto drop;
|
2018-12-13 21:36:57 +01:00
|
|
|
}
|
|
|
|
|
2019-05-29 11:47:12 +02:00
|
|
|
if (ntohs(udp_hdr->len) != (net_pkt_get_len(pkt) -
|
|
|
|
net_pkt_ip_hdr_len(pkt) -
|
2019-09-30 11:17:03 +02:00
|
|
|
net_pkt_ip_opts_len(pkt))) {
|
2019-05-29 11:47:12 +02:00
|
|
|
NET_DBG("DROP: Invalid hdr length");
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
|
2019-05-27 10:54:01 +02:00
|
|
|
if (IS_ENABLED(CONFIG_NET_UDP_CHECKSUM) &&
|
2023-10-30 13:23:02 +01:00
|
|
|
(net_if_need_calc_rx_checksum(net_pkt_iface(pkt)) ||
|
|
|
|
net_pkt_is_ip_reassembled(pkt))) {
|
2019-05-29 15:26:43 +02:00
|
|
|
if (!udp_hdr->chksum) {
|
|
|
|
if (IS_ENABLED(CONFIG_NET_UDP_MISSING_CHECKSUM) &&
|
|
|
|
net_pkt_family(pkt) == AF_INET) {
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
goto drop;
|
2019-05-27 10:54:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (net_calc_verify_chksum_udp(pkt) != 0U) {
|
|
|
|
NET_DBG("DROP: checksum mismatch");
|
|
|
|
goto drop;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
out:
|
2019-02-05 09:57:49 +01:00
|
|
|
return udp_hdr;
|
2018-12-13 21:36:57 +01:00
|
|
|
drop:
|
|
|
|
net_stats_update_udp_chkerr(net_pkt_iface(pkt));
|
|
|
|
return NULL;
|
|
|
|
}
|