/**************************************************************************** * net/utils/utils.h * * SPDX-License-Identifier: Apache-2.0 * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the * License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. * ****************************************************************************/ #ifndef __NET_UTILS_UTILS_H #define __NET_UTILS_UTILS_H /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ /* Some utils for port selection */ #define NET_PORT_RANDOM_INIT(port) \ do \ { \ arc4random_buf(&(port), sizeof(port)); \ (port) = (port) % (CONFIG_NET_DEFAULT_MAX_PORT - \ CONFIG_NET_DEFAULT_MIN_PORT + 1); \ (port) += CONFIG_NET_DEFAULT_MIN_PORT; \ } while (0) /* Get next net port number, and make sure that the port number is within * range. In host byte order. */ #define NET_PORT_NEXT_H(hport) \ do \ { \ ++(hport); \ if ((hport) > CONFIG_NET_DEFAULT_MAX_PORT || \ (hport) < CONFIG_NET_DEFAULT_MIN_PORT) \ { \ (hport) = CONFIG_NET_DEFAULT_MIN_PORT; \ } \ } while (0) /* Get next net port number, and make sure that the port number is within * range. In both network & host byte order. */ #define NET_PORT_NEXT_NH(nport, hport) \ do \ { \ NET_PORT_NEXT_H(hport); \ (nport) = HTONS(hport); \ } while (0) /* Network buffer pool related macros, in which: * pool: The name of the buffer pool * nodesize: The size of each node in the pool * prealloc: The number of pre-allocated buffers * dynalloc: The number per dynamic allocations * maxalloc: The number of max allocations, 0 means no limit */ #define NET_BUFPOOL_MAX(prealloc, dynalloc, maxalloc) \ (dynalloc) <= 0 ? (prealloc) : ((maxalloc) > 0 ? (maxalloc) : INT16_MAX) #define NET_BUFPOOL_DECLARE(pool, nodesize, prealloc, dynalloc, maxalloc) \ static char pool##_buffer[prealloc][nodesize]; \ static struct net_bufpool_s pool = \ { \ pool##_buffer[0], \ prealloc, \ dynalloc, \ -(int)(nodesize), \ SEM_INITIALIZER(NET_BUFPOOL_MAX(prealloc, dynalloc, maxalloc)), \ { NULL, NULL } \ }; #define NET_BUFPOOL_TIMEDALLOC(p,t) net_bufpool_timedalloc(&p, t) #define NET_BUFPOOL_TRYALLOC(p) net_bufpool_timedalloc(&p, 0) #define NET_BUFPOOL_ALLOC(p) net_bufpool_timedalloc(&p, UINT_MAX) #define NET_BUFPOOL_FREE(p,n) net_bufpool_free(&p, n) #define NET_BUFPOOL_TEST(p) net_bufpool_test(&p) /**************************************************************************** * Public Types ****************************************************************************/ /* These values control the behavior of net_timeval2desc */ enum tv2ds_remainder_e { TV2DS_TRUNC = 0, /* Truncate microsecond remainder */ TV2DS_ROUND, /* Round to the nearest full decisecond */ TV2DS_CEIL /* Force to next larger full decisecond */ }; /* This structure is used to manage a pool of network buffers */ struct net_bufpool_s { /* Allocation configuration */ FAR char *pool; /* The beginning of the pre-allocated buffer pool */ int prealloc; /* The number of pre-allocated buffers */ int dynalloc; /* The number per dynamic allocations */ int nodesize; /* The size of each node in the pool */ sem_t sem; /* The semaphore for waiting for free buffers */ sq_queue_t freebuffers; }; /**************************************************************************** * Public Data ****************************************************************************/ #ifdef __cplusplus #define EXTERN extern "C" extern "C" { #else #define EXTERN extern #endif /**************************************************************************** * Public Function Prototypes ****************************************************************************/ struct net_driver_s; /* Forward reference */ struct timeval; /* Forward reference */ /**************************************************************************** * Name: net_breaklock * * Description: * Break the lock, return information needed to restore re-entrant lock * state. * ****************************************************************************/ int net_breaklock(FAR unsigned int *count); /**************************************************************************** * Name: net_restorelock * * Description: * Restore the locked state * * Returned Value: * Zero (OK) is returned on success; a negated errno value is returned on * failured (probably -ECANCELED). * ****************************************************************************/ int net_restorelock(unsigned int count); /**************************************************************************** * Name: net_dsec2timeval * * Description: * Convert a decisecond value to a struct timeval. Needed by getsockopt() * to report timeout values. * * Input Parameters: * dsec The decisecond value to convert * tv The struct timeval to receive the converted value * * Returned Value: * None * * Assumptions: * ****************************************************************************/ void net_dsec2timeval(uint16_t dsec, FAR struct timeval *tv); /**************************************************************************** * Name: net_dsec2tick * * Description: * Convert a decisecond value to a system clock ticks. Used by IGMP logic. * * Input Parameters: * dsec The decisecond value to convert * * Returned Value: * The decisecond value expressed as system clock ticks * ****************************************************************************/ unsigned int net_dsec2tick(int dsec); /**************************************************************************** * Name: net_timeval2dsec * * Description: * Convert a struct timeval to deciseconds. Needed by setsockopt() to * save new timeout values. * * Input Parameters: * tv - The struct timeval to convert * remainder - Determines how to handler the microsecond remainder * * Returned Value: * The converted value * * Assumptions: * ****************************************************************************/ unsigned int net_timeval2dsec(FAR struct timeval *tv, enum tv2ds_remainder_e remainder); /**************************************************************************** * Name: net_ipv4_mask2pref * * Description: * Convert a 32-bit netmask to a prefix length. The NuttX IPv4 * networking uses 32-bit network masks internally. This function * converts the IPv4 netmask to a prefix length. * * The prefix length is the number of MS '1' bits on in the netmask. * This, of course, assumes that all MS bits are '1' and all LS bits are * '0' with no intermixed 1's and 0's. This function searches from the MS * bit until the first '0' is found (this does not necessary mean that * there might not be additional '1' bits following the firs '0', but that * will be a malformed netmask. * * Input Parameters: * mask An IPv4 netmask in the form of in_addr_t * * Returned Value: * The prefix length, range 0-32 on success; This function will not * fail. * ****************************************************************************/ #ifdef CONFIG_NET_IPv4 uint8_t net_ipv4_mask2pref(in_addr_t mask); #endif /**************************************************************************** * Name: net_ipv6_common_pref * * Description: * Calculate the common prefix length of two IPv6 addresses. * * Input Parameters: * a1,a2 Points to IPv6 addresses in the form of uint16_t[8] * * Returned Value: * The common prefix length, range 0-128 on success; This function will * not fail. * ****************************************************************************/ #ifdef CONFIG_NET_IPv6 uint8_t net_ipv6_common_pref(FAR const uint16_t *a1, FAR const uint16_t *a2); #endif /**************************************************************************** * Name: net_ipv6_mask2pref * * Description: * Convert a 128-bit netmask to a prefix length. The NuttX IPv6 * networking uses 128-bit network masks internally. This function * converts the IPv6 netmask to a prefix length. * * The prefix length is the number of MS '1' bits on in the netmask. * This, of course, assumes that all MS bits are '1' and all LS bits are * '0' with no intermixed 1's and 0's. This function searches from the MS * bit until the first '0' is found (this does not necessary mean that * there might not be additional '1' bits following the firs '0', but that * will be a malformed netmask. * * Input Parameters: * mask Points to an IPv6 netmask in the form of uint16_t[8] * * Returned Value: * The prefix length, range 0-128 on success; This function will not * fail. * ****************************************************************************/ #ifdef CONFIG_NET_IPv6 uint8_t net_ipv6_mask2pref(FAR const uint16_t *mask); #endif /**************************************************************************** * Name: net_ipv6_pref2mask * * Description: * Convert a IPv6 prefix length to a network mask. The prefix length * specifies the number of MS bits under mask (0-128) * * Input Parameters: * mask - The location to return the netmask. * preflen - Determines the width of the netmask (in bits). Range 0-128 * * Returned Value: * None * ****************************************************************************/ #ifdef CONFIG_NET_IPv6 void net_ipv6_pref2mask(net_ipv6addr_t mask, uint8_t preflen); #endif /**************************************************************************** * Name: net_ipv6_payload * * Description: * Given a pointer to the IPv6 header, this function will return a pointer * to the beginning of the L4 payload. * * Input Parameters: * ipv6 - A pointer to the IPv6 header. * proto - The location to return the protocol number in the IPv6 header. * * Returned Value: * A pointer to the beginning of the payload. * ****************************************************************************/ #ifdef CONFIG_NET_IPv6 FAR void *net_ipv6_payload(FAR struct ipv6_hdr_s *ipv6, FAR uint8_t *proto); #endif /**************************************************************************** * Name: net_iob_concat * * Description: * Concatenate iob_s chain iob2 to iob1, if CONFIG_NET_RECV_PACK is * endabled, pack all data in the I/O buffer chain. * * Returned Value: * The number of bytes actually buffered is returned. This will be either * zero or equal to iob1->io_pktlen. * ****************************************************************************/ #ifdef CONFIG_MM_IOB uint16_t net_iob_concat(FAR struct iob_s **iob1, FAR struct iob_s **iob2); #endif /**************************************************************************** * Name: net_bufpool_timedalloc * * Description: * Allocate a buffer from the pool. If no buffer is available, then wait * for the specified timeout. * * Input Parameters: * pool - The pool from which to allocate the buffer * timeout - The maximum time to wait for a buffer to become available. * * Returned Value: * A reference to the allocated buffer, which is guaranteed to be zeroed. * NULL is returned on a timeout. * ****************************************************************************/ FAR void *net_bufpool_timedalloc(FAR struct net_bufpool_s *pool, unsigned int timeout); /**************************************************************************** * Name: net_bufpool_free * * Description: * Free a buffer from the pool. * * Input Parameters: * pool - The pool from which to allocate the buffer * node - The buffer to be freed * ****************************************************************************/ void net_bufpool_free(FAR struct net_bufpool_s *pool, FAR void *node); /**************************************************************************** * Name: net_bufpool_test * * Description: * Check if there is room in the buffer pool. Does not reserve any space. * * Assumptions: * None. * ****************************************************************************/ int net_bufpool_test(FAR struct net_bufpool_s *pool); /**************************************************************************** * Name: net_chksum_adjust * * Description: * Adjusts the checksum of a packet without having to completely * recalculate it, as described in RFC 3022, Section 4.2, Page 9. * * Input Parameters: * chksum - points to the chksum in the packet * optr - points to the old data in the packet * olen - length of old data * nptr - points to the new data in the packet * nlen - length of new data * * Limitations: * The algorithm is applicable only for even offsets and even lengths. * ****************************************************************************/ void net_chksum_adjust(FAR uint16_t *chksum, FAR const uint16_t *optr, ssize_t olen, FAR const uint16_t *nptr, ssize_t nlen); /**************************************************************************** * Name: tcp_chksum, tcp_ipv4_chksum, and tcp_ipv6_chksum * * Description: * Calculate the TCP checksum of the packet in d_buf and d_appdata. * * The TCP checksum is the Internet checksum of data contents of the * TCP segment, and a pseudo-header as defined in RFC793. * * Note: The d_appdata pointer that points to the packet data may * point anywhere in memory, so it is not possible to simply calculate * the Internet checksum of the contents of the d_buf buffer. * * Returned Value: * The TCP checksum of the TCP segment in d_buf and pointed to by * d_appdata. * ****************************************************************************/ #ifdef CONFIG_NET_IPv4 uint16_t tcp_ipv4_chksum(FAR struct net_driver_s *dev); #endif #ifdef CONFIG_NET_IPv6 uint16_t tcp_ipv6_chksum(FAR struct net_driver_s *dev); #endif #if defined(CONFIG_NET_IPv4) && defined(CONFIG_NET_IPv6) uint16_t tcp_chksum(FAR struct net_driver_s *dev); #elif defined(CONFIG_NET_IPv4) # define tcp_chksum(d) tcp_ipv4_chksum(d) #else /* if defined(CONFIG_NET_IPv6) */ # define tcp_chksum(d) tcp_ipv6_chksum(d) #endif /**************************************************************************** * Name: udp_ipv4_chksum * * Description: * Calculate the UDP/IPv4 checksum of the packet in d_buf and d_appdata. * ****************************************************************************/ #if defined(CONFIG_NET_UDP_CHECKSUMS) && defined(CONFIG_NET_IPv4) uint16_t udp_ipv4_chksum(FAR struct net_driver_s *dev); #endif /**************************************************************************** * Name: udp_ipv6_chksum * * Description: * Calculate the UDP/IPv6 checksum of the packet in d_buf and d_appdata. * ****************************************************************************/ #if defined(CONFIG_NET_UDP_CHECKSUMS) && defined(CONFIG_NET_IPv6) uint16_t udp_ipv6_chksum(FAR struct net_driver_s *dev); #endif /**************************************************************************** * Name: icmp_chksum * * Description: * Calculate the checksum of the IPv4 ICMP message * ****************************************************************************/ #ifdef CONFIG_NET_ICMP uint16_t icmp_chksum(FAR struct net_driver_s *dev, int len); #endif /**************************************************************************** * Name: icmp_chksum_iob * * Description: * Calculate the checksum of the ICMP message, the input can be an I/O * buffer chain * ****************************************************************************/ #if defined(CONFIG_NET_ICMP) && defined(CONFIG_MM_IOB) uint16_t icmp_chksum_iob(FAR struct iob_s *iob); #endif /* CONFIG_NET_ICMP && CONFIG_MM_IOB */ /**************************************************************************** * Name: icmpv6_chksum * * Description: * Calculate the checksum of the ICMPv6 message * ****************************************************************************/ #ifdef CONFIG_NET_ICMPv6 uint16_t icmpv6_chksum(FAR struct net_driver_s *dev, unsigned int iplen); #endif /**************************************************************************** * Name: cmsg_append * * Description: * Append specified data into the control message, msg_control and * msg_controllen will be changed to the appropriate value when success * * Input Parameters: * msg - Buffer to receive the message. * level - The level of control message. * type - The type of control message. * value - If the value is not NULL, this interface copies the data * to the appropriate location in msg_control, and modify * msg_control and msg_controllen. * If the value is NULL, just modify the corresponding value * of msg. * value_len - The value length of control message. * * Returned Value: * On success, a pointer to the start address of control message data, * the caller can copy the data in. * On failure, return NULL, because of msg_controllen is not enough * ****************************************************************************/ FAR void *cmsg_append(FAR struct msghdr *msg, int level, int type, FAR void *value, int value_len); #undef EXTERN #ifdef __cplusplus } #endif #endif /* __NET_UTILS_UTILS_H */