/* libhw/generic/net.h - Device-independent network definitions * * Copyright (C) 2024-2025 Luke T. Shumaker * SPDX-License-Identifier: AGPL-3.0-or-later */ #ifndef _LIBHW_GENERIC_NET_H_ #define _LIBHW_GENERIC_NET_H_ #include /* for PRI{u,x}{n} */ #include /* for bool */ #include /* for size_t */ #include /* for uint{n}_t} */ #include /* for ssize_t */ #include /* Errnos *********************************************************************/ #define NET_EOTHER 1 #define NET_EARP_TIMEOUT 2 #define NET_EACK_TIMEOUT 3 #define NET_ERECV_TIMEOUT 4 #define NET_ETHREAD 5 #define NET_ECLOSED 6 #define NET_EMSGSIZE 7 const char *net_strerror(int net_errno); /* Address types **************************************************************/ struct net_ip4_addr { unsigned char octets[4]; }; static const struct net_ip4_addr net_ip4_addr_broadcast = {{255, 255, 255, 255}}; static const struct net_ip4_addr net_ip4_addr_zero = {{0, 0, 0, 0}}; #define PRI_net_ip4_addr "%"PRIu8".%"PRIu8".%"PRIu8".%"PRIu8 #define ARG_net_ip4_addr(addr) (addr).octets[0], \ (addr).octets[1], \ (addr).octets[2], \ (addr).octets[3] struct net_eth_addr { unsigned char octets[6]; }; #define PRI_net_eth_addr "%02"PRIx8":%02"PRIx8":%02"PRIx8":%02"PRIx8":%02"PRIx8":%02"PRIx8 #define ARG_net_eth_addr(addr) (addr).octets[0], \ (addr).octets[1], \ (addr).octets[2], \ (addr).octets[3], \ (addr).octets[4], \ (addr).octets[5] /* Streams (e.g. TCP) *********************************************************/ lo_interface net_stream_conn; #define net_stream_listener_LO_IFACE \ /** \ * It is invalid to accept() a new connection if an existing \ * connection is still open. \ */ \ LO_FUNC(lo_interface net_stream_conn, accept) \ \ /** \ * The net_stream_conn returned from accept() may still be \ * valid after the listener is closed. \ */ \ LO_NEST(io_closer) LO_INTERFACE(net_stream_listener) #define net_stream_conn_LO_IFACE \ LO_NEST(io_readwriter) \ LO_NEST(io_bidi_closer) \ \ /** \ * Set a timestamp after which calls to read() will return \ * NET_ETIMEDOUT. The timestamp is in nanoseconds on the \ * system monotonic clock, which is usually (on pico-sdk and \ * on the Linux kernel) nanoseconds-since-boot. \ * \ * A zero value disables the deadline. \ * \ * (2⁶⁴-1 nanoseconds is more than 500 years; there is little \ * risk of this overflowing) \ */ \ LO_FUNC(void, set_read_deadline, uint64_t ns_since_boot) LO_INTERFACE(net_stream_conn) /* Packets (e.g. UDP) *********************************************************/ #define net_packet_conn_LO_IFACE \ LO_FUNC(ssize_t, sendto, \ void *buf, size_t len, \ struct net_ip4_addr node, uint16_t port) \ \ /** \ * @return The full length of the message, which may be more \ * than the given `len` (as if the Linux MSG_TRUNC flag were \ * given). \ */ \ LO_FUNC(ssize_t, recvfrom, \ void *buf, size_t len, \ struct net_ip4_addr *ret_node, uint16_t *ret_port) \ \ /** \ * Set a timestamp after which calls to recvfrom() will return \ * NET_ETIMEDOUT. The timestamp is in nanoseconds on the \ * system monotonic clock, which is usually (on pico-sdk and \ * on the Linux kernel) nanoseconds-since-boot. \ * \ * A zero value disables the deadline. \ * \ * (2⁶⁴-1 nanoseconds is more than 500 years; there is little \ * risk of this overflowing) \ */ \ LO_FUNC(void, set_recv_deadline, \ uint64_t ns_since_boot) \ \ LO_NEST(io_closer) LO_INTERFACE(net_packet_conn) /* Interfaces *****************************************************************/ struct net_iface_config { struct net_ip4_addr addr; struct net_ip4_addr gateway_addr; struct net_ip4_addr subnet_mask; }; #define net_iface_LO_IFACE \ LO_FUNC(struct net_eth_addr , hwaddr ) \ LO_FUNC(void , ifup , struct net_iface_config) \ LO_FUNC(void , ifdown ) \ \ LO_FUNC(lo_interface net_stream_listener, tcp_listen, uint16_t local_port) \ LO_FUNC(lo_interface net_stream_conn , tcp_dial , struct net_ip4_addr remote_node, uint16_t remote_port) \ LO_FUNC(lo_interface net_packet_conn , udp_conn , uint16_t local_port) LO_INTERFACE(net_iface) #endif /* _LIBHW_GENERIC_NET_H_ */