/** * Copyright (c) 2023-2024 Marcin Niestroj * * SPDX-License-Identifier: Apache-2.0 */ /** * @file * * Linux (bottom) side of NSOS (Native Simulator Offloaded Sockets). */ #define _DEFAULT_SOURCE #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "nsos.h" #include "nsos_errno.h" #include "nsos_fcntl.h" #include "nsos_netdb.h" #include "nsos_socket.h" #include "board_soc.h" #include "irq_ctrl.h" #include "nsi_hws_models_if.h" #include "nsi_tasks.h" #include "nsi_tracing.h" #include static int nsos_epoll_fd; static int nsos_adapt_nfds; #ifndef ARRAY_SIZE #define ARRAY_SIZE(array) (sizeof(array) / sizeof((array)[0])) #endif #ifndef CONTAINER_OF #define CONTAINER_OF(ptr, type, field) \ ((type *)(((char *)(ptr)) - offsetof(type, field))) #endif int nsos_adapt_get_errno(void) { return errno_to_nsos_mid(errno); } static int socket_family_from_nsos_mid(int family_mid, int *family) { switch (family_mid) { case NSOS_MID_AF_UNSPEC: *family = AF_UNSPEC; break; case NSOS_MID_AF_INET: *family = AF_INET; break; case NSOS_MID_AF_INET6: *family = AF_INET6; break; case NSOS_MID_AF_UNIX: *family = AF_UNIX; break; case NSOS_MID_AF_PACKET: *family = AF_PACKET; break; default: nsi_print_warning("%s: socket family %d not supported\n", __func__, family_mid); return -NSOS_MID_EAFNOSUPPORT; } return 0; } static int socket_family_to_nsos_mid(int family, int *family_mid) { switch (family) { case AF_UNSPEC: *family_mid = NSOS_MID_AF_UNSPEC; break; case AF_INET: *family_mid = NSOS_MID_AF_INET; break; case AF_INET6: *family_mid = NSOS_MID_AF_INET6; break; case AF_UNIX: *family_mid = NSOS_MID_AF_UNIX; break; case AF_PACKET: *family_mid = NSOS_MID_AF_PACKET; break; default: nsi_print_warning("%s: socket family %d not supported\n", __func__, family); return -NSOS_MID_EAFNOSUPPORT; } return 0; } static int socket_proto_from_nsos_mid(int proto_mid, int *proto) { switch (proto_mid) { case NSOS_MID_IPPROTO_IP: *proto = IPPROTO_IP; break; case NSOS_MID_IPPROTO_ICMP: *proto = IPPROTO_ICMP; break; case NSOS_MID_IPPROTO_IGMP: *proto = IPPROTO_IGMP; break; case NSOS_MID_IPPROTO_IPIP: *proto = IPPROTO_IPIP; break; case NSOS_MID_IPPROTO_TCP: *proto = IPPROTO_TCP; break; case NSOS_MID_IPPROTO_UDP: *proto = IPPROTO_UDP; break; case NSOS_MID_IPPROTO_IPV6: *proto = IPPROTO_IPV6; break; case NSOS_MID_IPPROTO_RAW: *proto = IPPROTO_RAW; break; case NSOS_MID_IPPROTO_ETH_P_ALL: *proto = htons(ETH_P_ALL); break; default: nsi_print_warning("%s: socket protocol %d not supported\n", __func__, proto_mid); return -NSOS_MID_EPROTONOSUPPORT; } return 0; } static int socket_proto_to_nsos_mid(int proto, int *proto_mid) { switch (proto) { case IPPROTO_IP: *proto_mid = NSOS_MID_IPPROTO_IP; break; case IPPROTO_ICMP: *proto_mid = NSOS_MID_IPPROTO_ICMP; break; case IPPROTO_IGMP: *proto_mid = NSOS_MID_IPPROTO_IGMP; break; case IPPROTO_IPIP: *proto_mid = NSOS_MID_IPPROTO_IPIP; break; case IPPROTO_TCP: *proto_mid = NSOS_MID_IPPROTO_TCP; break; case IPPROTO_UDP: *proto_mid = NSOS_MID_IPPROTO_UDP; break; case IPPROTO_IPV6: *proto_mid = NSOS_MID_IPPROTO_IPV6; break; case IPPROTO_RAW: *proto_mid = NSOS_MID_IPPROTO_RAW; break; case ETH_P_ALL: *proto_mid = htons(NSOS_MID_IPPROTO_ETH_P_ALL); break; default: nsi_print_warning("%s: socket protocol %d not supported\n", __func__, proto); return -NSOS_MID_EPROTONOSUPPORT; } return 0; } static int socket_type_from_nsos_mid(int type_mid, int *type) { switch (type_mid) { case NSOS_MID_SOCK_STREAM: *type = SOCK_STREAM; break; case NSOS_MID_SOCK_DGRAM: *type = SOCK_DGRAM; break; case NSOS_MID_SOCK_RAW: *type = SOCK_RAW; break; default: nsi_print_warning("%s: socket type %d not supported\n", __func__, type_mid); return -NSOS_MID_ESOCKTNOSUPPORT; } return 0; } static int socket_type_to_nsos_mid(int type, int *type_mid) { switch (type) { case SOCK_STREAM: *type_mid = NSOS_MID_SOCK_STREAM; break; case SOCK_DGRAM: *type_mid = NSOS_MID_SOCK_DGRAM; break; case SOCK_RAW: *type_mid = NSOS_MID_SOCK_RAW; break; default: nsi_print_warning("%s: socket type %d not supported\n", __func__, type); return -NSOS_MID_ESOCKTNOSUPPORT; } return 0; } static int socket_flags_from_nsos_mid(int flags_mid) { int flags = 0; nsos_socket_flag_convert(&flags_mid, NSOS_MID_MSG_PEEK, &flags, MSG_PEEK); nsos_socket_flag_convert(&flags_mid, NSOS_MID_MSG_TRUNC, &flags, MSG_TRUNC); nsos_socket_flag_convert(&flags_mid, NSOS_MID_MSG_DONTWAIT, &flags, MSG_DONTWAIT); nsos_socket_flag_convert(&flags_mid, NSOS_MID_MSG_WAITALL, &flags, MSG_WAITALL); if (flags_mid != 0) { return -NSOS_MID_EINVAL; } return flags; } int nsos_adapt_socket(int family_mid, int type_mid, int proto_mid) { int family; int type; int proto; int ret; ret = socket_family_from_nsos_mid(family_mid, &family); if (ret < 0) { return ret; } ret = socket_type_from_nsos_mid(type_mid, &type); if (ret < 0) { return ret; } ret = socket_proto_from_nsos_mid(proto_mid, &proto); if (ret < 0) { return ret; } ret = socket(family, type, proto); if (ret < 0) { return -errno_to_nsos_mid(errno); } return ret; } static int sockaddr_from_nsos_mid(struct sockaddr **addr, socklen_t *addrlen, const struct nsos_mid_sockaddr *addr_mid, size_t addrlen_mid) { if (!addr_mid || addrlen_mid == 0) { *addr = NULL; *addrlen = 0; return 0; } switch (addr_mid->sa_family) { case NSOS_MID_AF_INET: { const struct nsos_mid_sockaddr_in *addr_in_mid = (const struct nsos_mid_sockaddr_in *)addr_mid; struct sockaddr_in *addr_in = (struct sockaddr_in *)*addr; addr_in->sin_family = AF_INET; addr_in->sin_port = addr_in_mid->sin_port; addr_in->sin_addr.s_addr = addr_in_mid->sin_addr; *addrlen = sizeof(*addr_in); return 0; } case NSOS_MID_AF_INET6: { const struct nsos_mid_sockaddr_in6 *addr_in_mid = (const struct nsos_mid_sockaddr_in6 *)addr_mid; struct sockaddr_in6 *addr_in = (struct sockaddr_in6 *)*addr; addr_in->sin6_family = AF_INET6; addr_in->sin6_port = addr_in_mid->sin6_port; addr_in->sin6_flowinfo = 0; memcpy(addr_in->sin6_addr.s6_addr, addr_in_mid->sin6_addr, sizeof(addr_in->sin6_addr.s6_addr)); addr_in->sin6_scope_id = addr_in_mid->sin6_scope_id; *addrlen = sizeof(*addr_in); return 0; } case NSOS_MID_AF_UNIX: { const struct nsos_mid_sockaddr_un *addr_un_mid = (const struct nsos_mid_sockaddr_un *)addr_mid; struct sockaddr_un *addr_un = (struct sockaddr_un *)*addr; addr_un->sun_family = AF_UNIX; memcpy(addr_un->sun_path, addr_un_mid->sun_path, sizeof(addr_un->sun_path)); *addrlen = sizeof(*addr_un); return 0; } case NSOS_MID_AF_PACKET: { const struct nsos_mid_sockaddr_ll *addr_ll_mid = (const struct nsos_mid_sockaddr_ll *)addr_mid; struct sockaddr_ll *addr_ll = (struct sockaddr_ll *)*addr; addr_ll->sll_family = NSOS_MID_AF_PACKET; addr_ll->sll_protocol = addr_ll_mid->sll_protocol; addr_ll->sll_ifindex = addr_ll_mid->sll_ifindex; addr_ll->sll_hatype = addr_ll_mid->sll_hatype; addr_ll->sll_pkttype = addr_ll_mid->sll_pkttype; addr_ll->sll_halen = addr_ll_mid->sll_halen; memcpy(addr_ll->sll_addr, addr_ll_mid->sll_addr, sizeof(addr_ll->sll_addr)); *addrlen = sizeof(*addr_ll); return 0; } } return -NSOS_MID_EINVAL; } static int sockaddr_to_nsos_mid(const struct sockaddr *addr, socklen_t addrlen, struct nsos_mid_sockaddr *addr_mid, size_t *addrlen_mid) { if (!addr || addrlen == 0) { *addrlen_mid = 0; return 0; } switch (addr->sa_family) { case AF_INET: { struct nsos_mid_sockaddr_in *addr_in_mid = (struct nsos_mid_sockaddr_in *)addr_mid; const struct sockaddr_in *addr_in = (const struct sockaddr_in *)addr; if (addr_in_mid) { addr_in_mid->sin_family = NSOS_MID_AF_INET; addr_in_mid->sin_port = addr_in->sin_port; addr_in_mid->sin_addr = addr_in->sin_addr.s_addr; } if (addrlen_mid) { *addrlen_mid = sizeof(*addr_in); } return 0; } case AF_INET6: { struct nsos_mid_sockaddr_in6 *addr_in_mid = (struct nsos_mid_sockaddr_in6 *)addr_mid; const struct sockaddr_in6 *addr_in = (const struct sockaddr_in6 *)addr; if (addr_in_mid) { addr_in_mid->sin6_family = NSOS_MID_AF_INET6; addr_in_mid->sin6_port = addr_in->sin6_port; memcpy(addr_in_mid->sin6_addr, addr_in->sin6_addr.s6_addr, sizeof(addr_in_mid->sin6_addr)); addr_in_mid->sin6_scope_id = addr_in->sin6_scope_id; } if (addrlen_mid) { *addrlen_mid = sizeof(*addr_in); } return 0; } case AF_UNIX: { struct nsos_mid_sockaddr_un *addr_un_mid = (struct nsos_mid_sockaddr_un *)addr_mid; const struct sockaddr_un *addr_un = (const struct sockaddr_un *)addr; if (addr_un_mid) { addr_un_mid->sun_family = NSOS_MID_AF_UNIX; memcpy(addr_un_mid->sun_path, addr_un->sun_path, sizeof(addr_un_mid->sun_path)); } if (addrlen_mid) { *addrlen_mid = sizeof(*addr_un); } return 0; } case AF_PACKET: { struct nsos_mid_sockaddr_ll *addr_ll_mid = (struct nsos_mid_sockaddr_ll *)addr_mid; const struct sockaddr_ll *addr_ll = (const struct sockaddr_ll *)addr; if (addr_ll_mid) { addr_ll_mid->sll_family = NSOS_MID_AF_PACKET; addr_ll_mid->sll_protocol = addr_ll->sll_protocol; addr_ll_mid->sll_ifindex = addr_ll->sll_ifindex; } if (addrlen_mid) { *addrlen_mid = sizeof(*addr_ll); } return 0; } } nsi_print_warning("%s: socket family %d not supported\n", __func__, addr->sa_family); return -NSOS_MID_EINVAL; } int nsos_adapt_bind(int fd, const struct nsos_mid_sockaddr *addr_mid, size_t addrlen_mid) { struct sockaddr_storage addr_storage; struct sockaddr *addr = (struct sockaddr *)&addr_storage; socklen_t addrlen; int ret; ret = sockaddr_from_nsos_mid(&addr, &addrlen, addr_mid, addrlen_mid); if (ret < 0) { return ret; } ret = bind(fd, addr, addrlen); if (ret < 0) { return -errno_to_nsos_mid(errno); } return ret; } int nsos_adapt_connect(int fd, const struct nsos_mid_sockaddr *addr_mid, size_t addrlen_mid) { struct sockaddr_storage addr_storage; struct sockaddr *addr = (struct sockaddr *)&addr_storage; socklen_t addrlen; int ret; ret = sockaddr_from_nsos_mid(&addr, &addrlen, addr_mid, addrlen_mid); if (ret < 0) { return ret; } ret = connect(fd, addr, addrlen); if (ret < 0) { return -errno_to_nsos_mid(errno); } return ret; } int nsos_adapt_listen(int fd, int backlog) { int ret; ret = listen(fd, backlog); if (ret < 0) { return -errno_to_nsos_mid(errno); } return ret; } int nsos_adapt_accept(int fd, struct nsos_mid_sockaddr *addr_mid, size_t *addrlen_mid) { struct sockaddr_storage addr_storage; struct sockaddr *addr = (struct sockaddr *)&addr_storage; socklen_t addrlen = sizeof(addr_storage); int ret; int err; ret = accept(fd, addr, &addrlen); if (ret < 0) { return -errno_to_nsos_mid(errno); } err = sockaddr_to_nsos_mid(addr, addrlen, addr_mid, addrlen_mid); if (err) { close(ret); return err; } return ret; } int nsos_adapt_sendto(int fd, const void *buf, size_t len, int flags, const struct nsos_mid_sockaddr *addr_mid, size_t addrlen_mid) { struct sockaddr_storage addr_storage; struct sockaddr *addr = (struct sockaddr *)&addr_storage; socklen_t addrlen; int ret; ret = sockaddr_from_nsos_mid(&addr, &addrlen, addr_mid, addrlen_mid); if (ret < 0) { return ret; } ret = sendto(fd, buf, len, socket_flags_from_nsos_mid(flags) | MSG_NOSIGNAL, addr, addrlen); if (ret < 0) { return -errno_to_nsos_mid(errno); } return ret; } int nsos_adapt_sendmsg(int fd, const struct nsos_mid_msghdr *msg_mid, int flags) { struct sockaddr_storage addr_storage; struct sockaddr *addr = (struct sockaddr *)&addr_storage; struct msghdr msg; struct iovec *msg_iov; socklen_t addrlen; int ret; ret = sockaddr_from_nsos_mid(&addr, &addrlen, msg_mid->msg_name, msg_mid->msg_namelen); if (ret < 0) { return ret; } msg_iov = calloc(msg_mid->msg_iovlen, sizeof(*msg_iov)); if (!msg_iov) { ret = -ENOMEM; return ret; } for (size_t i = 0; i < msg_mid->msg_iovlen; i++) { msg_iov[i].iov_base = msg_mid->msg_iov[i].iov_base; msg_iov[i].iov_len = msg_mid->msg_iov[i].iov_len; } msg.msg_name = addr; msg.msg_namelen = addrlen; msg.msg_iov = msg_iov; msg.msg_iovlen = msg_mid->msg_iovlen; msg.msg_control = NULL; msg.msg_controllen = 0; msg.msg_flags = 0; ret = sendmsg(fd, &msg, socket_flags_from_nsos_mid(flags) | MSG_NOSIGNAL); if (ret < 0) { ret = -errno_to_nsos_mid(errno); } free(msg_iov); return ret; } int nsos_adapt_recvfrom(int fd, void *buf, size_t len, int flags, struct nsos_mid_sockaddr *addr_mid, size_t *addrlen_mid) { struct sockaddr_storage addr_storage; struct sockaddr *addr = (struct sockaddr *)&addr_storage; socklen_t addrlen = sizeof(addr_storage); int ret; int err; ret = recvfrom(fd, buf, len, socket_flags_from_nsos_mid(flags), addr, &addrlen); if (ret < 0) { return -errno_to_nsos_mid(errno); } err = sockaddr_to_nsos_mid(addr, addrlen, addr_mid, addrlen_mid); if (err) { return err; } return ret; } static int nsos_adapt_getsockopt_int(int fd, int level, int optname, void *optval, size_t *nsos_mid_optlen) { socklen_t optlen = *nsos_mid_optlen; int ret; ret = getsockopt(fd, level, optname, optval, &optlen); if (ret < 0) { return -errno_to_nsos_mid(errno); } *nsos_mid_optlen = optlen; return 0; } int nsos_adapt_getsockopt(int fd, int nsos_mid_level, int nsos_mid_optname, void *nsos_mid_optval, size_t *nsos_mid_optlen) { switch (nsos_mid_level) { case NSOS_MID_SOL_SOCKET: switch (nsos_mid_optname) { case NSOS_MID_SO_ERROR: { int err; socklen_t optlen = sizeof(err); int ret; ret = getsockopt(fd, SOL_SOCKET, SO_ERROR, &err, &optlen); if (ret < 0) { return -errno_to_nsos_mid(errno); } *(int *)nsos_mid_optval = errno_to_nsos_mid(err); return 0; } case NSOS_MID_SO_TYPE: { int type; socklen_t optlen = sizeof(type); int ret; int err; ret = getsockopt(fd, SOL_SOCKET, SO_TYPE, &type, &optlen); if (ret < 0) { return -errno_to_nsos_mid(errno); } err = socket_type_to_nsos_mid(type, nsos_mid_optval); if (err) { return err; } return 0; } case NSOS_MID_SO_PROTOCOL: { int proto; socklen_t optlen = sizeof(proto); int ret; int err; ret = getsockopt(fd, SOL_SOCKET, SO_PROTOCOL, &proto, &optlen); if (ret < 0) { return -errno_to_nsos_mid(errno); } err = socket_proto_to_nsos_mid(proto, nsos_mid_optval); if (err) { return err; } return 0; } case NSOS_MID_SO_DOMAIN: { int family; socklen_t optlen = sizeof(family); int ret; int err; ret = getsockopt(fd, SOL_SOCKET, SO_DOMAIN, &family, &optlen); if (ret < 0) { return -errno_to_nsos_mid(errno); } err = socket_family_to_nsos_mid(family, nsos_mid_optval); if (err) { return err; } return 0; } case NSOS_MID_SO_RCVBUF: return nsos_adapt_getsockopt_int(fd, SOL_SOCKET, SO_RCVBUF, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_SO_SNDBUF: return nsos_adapt_getsockopt_int(fd, SOL_SOCKET, SO_SNDBUF, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_SO_REUSEADDR: return nsos_adapt_getsockopt_int(fd, SOL_SOCKET, SO_REUSEADDR, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_SO_REUSEPORT: return nsos_adapt_getsockopt_int(fd, SOL_SOCKET, SO_REUSEPORT, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_SO_LINGER: return nsos_adapt_getsockopt_int(fd, SOL_SOCKET, SO_LINGER, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_SO_KEEPALIVE: return nsos_adapt_getsockopt_int(fd, SOL_SOCKET, SO_KEEPALIVE, nsos_mid_optval, nsos_mid_optlen); } break; case NSOS_MID_IPPROTO_TCP: switch (nsos_mid_optname) { case NSOS_MID_TCP_NODELAY: return nsos_adapt_getsockopt_int(fd, IPPROTO_TCP, TCP_NODELAY, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_TCP_KEEPIDLE: return nsos_adapt_getsockopt_int(fd, IPPROTO_TCP, TCP_KEEPIDLE, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_TCP_KEEPINTVL: return nsos_adapt_getsockopt_int(fd, IPPROTO_TCP, TCP_KEEPINTVL, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_TCP_KEEPCNT: return nsos_adapt_getsockopt_int(fd, IPPROTO_TCP, TCP_KEEPCNT, nsos_mid_optval, nsos_mid_optlen); } break; case NSOS_MID_IPPROTO_IPV6: switch (nsos_mid_optname) { case NSOS_MID_IPV6_V6ONLY: return nsos_adapt_getsockopt_int(fd, IPPROTO_IPV6, IPV6_V6ONLY, nsos_mid_optval, nsos_mid_optlen); } break; } return -NSOS_MID_EOPNOTSUPP; } static int nsos_adapt_setsockopt_int(int fd, int level, int optname, const void *optval, size_t optlen) { int ret; ret = setsockopt(fd, level, optname, optval, optlen); if (ret < 0) { return -errno_to_nsos_mid(errno); } return 0; } int nsos_adapt_setsockopt(int fd, int nsos_mid_level, int nsos_mid_optname, const void *nsos_mid_optval, size_t nsos_mid_optlen) { switch (nsos_mid_level) { case NSOS_MID_SOL_SOCKET: switch (nsos_mid_optname) { case NSOS_MID_SO_PRIORITY: return nsos_adapt_setsockopt_int(fd, SOL_SOCKET, SO_PRIORITY, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_SO_RCVTIMEO: { const struct nsos_mid_timeval *nsos_mid_tv = nsos_mid_optval; struct timeval tv = { .tv_sec = nsos_mid_tv->tv_sec, .tv_usec = nsos_mid_tv->tv_usec, }; int ret; ret = setsockopt(fd, SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(tv)); if (ret < 0) { return -errno_to_nsos_mid(errno); } return 0; } case NSOS_MID_SO_SNDTIMEO: { const struct nsos_mid_timeval *nsos_mid_tv = nsos_mid_optval; struct timeval tv = { .tv_sec = nsos_mid_tv->tv_sec, .tv_usec = nsos_mid_tv->tv_usec, }; int ret; ret = setsockopt(fd, SOL_SOCKET, SO_SNDTIMEO, &tv, sizeof(tv)); if (ret < 0) { return -errno_to_nsos_mid(errno); } return 0; } case NSOS_MID_SO_RCVBUF: return nsos_adapt_setsockopt_int(fd, SOL_SOCKET, SO_RCVBUF, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_SO_SNDBUF: return nsos_adapt_setsockopt_int(fd, SOL_SOCKET, SO_SNDBUF, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_SO_REUSEADDR: return nsos_adapt_setsockopt_int(fd, SOL_SOCKET, SO_REUSEADDR, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_SO_REUSEPORT: return nsos_adapt_setsockopt_int(fd, SOL_SOCKET, SO_REUSEPORT, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_SO_LINGER: return nsos_adapt_setsockopt_int(fd, SOL_SOCKET, SO_LINGER, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_SO_KEEPALIVE: return nsos_adapt_setsockopt_int(fd, SOL_SOCKET, SO_KEEPALIVE, nsos_mid_optval, nsos_mid_optlen); } break; case NSOS_MID_IPPROTO_TCP: switch (nsos_mid_optname) { case NSOS_MID_TCP_NODELAY: return nsos_adapt_setsockopt_int(fd, IPPROTO_TCP, TCP_NODELAY, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_TCP_KEEPIDLE: return nsos_adapt_setsockopt_int(fd, IPPROTO_TCP, TCP_KEEPIDLE, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_TCP_KEEPINTVL: return nsos_adapt_setsockopt_int(fd, IPPROTO_TCP, TCP_KEEPINTVL, nsos_mid_optval, nsos_mid_optlen); case NSOS_MID_TCP_KEEPCNT: return nsos_adapt_setsockopt_int(fd, IPPROTO_TCP, TCP_KEEPCNT, nsos_mid_optval, nsos_mid_optlen); } break; case NSOS_MID_IPPROTO_IPV6: switch (nsos_mid_optname) { case NSOS_MID_IPV6_V6ONLY: return nsos_adapt_setsockopt_int(fd, IPPROTO_IPV6, IPV6_V6ONLY, nsos_mid_optval, nsos_mid_optlen); } break; } return -NSOS_MID_EOPNOTSUPP; } #define MAP_POLL_EPOLL(_event_from, _event_to) \ if (events_from & (_event_from)) { \ events_from &= ~(_event_from); \ events_to |= _event_to; \ } static int nsos_poll_to_epoll_events(int events_from) { int events_to = 0; MAP_POLL_EPOLL(POLLIN, EPOLLIN); MAP_POLL_EPOLL(POLLOUT, EPOLLOUT); MAP_POLL_EPOLL(POLLERR, EPOLLERR); MAP_POLL_EPOLL(POLLHUP, EPOLLHUP); return events_to; } static int nsos_epoll_to_poll_events(int events_from) { int events_to = 0; MAP_POLL_EPOLL(EPOLLIN, POLLIN); MAP_POLL_EPOLL(EPOLLOUT, POLLOUT); MAP_POLL_EPOLL(EPOLLERR, POLLERR); MAP_POLL_EPOLL(EPOLLHUP, POLLHUP); return events_to; } #undef MAP_POLL_EPOLL static uint64_t nsos_adapt_poll_time = NSI_NEVER; void nsos_adapt_poll_add(struct nsos_mid_pollfd *pollfd) { struct epoll_event ev = { .data.ptr = pollfd, .events = nsos_poll_to_epoll_events(pollfd->events), }; int err; nsos_adapt_nfds++; err = epoll_ctl(nsos_epoll_fd, EPOLL_CTL_ADD, pollfd->fd, &ev); if (err) { nsi_print_error_and_exit("error in EPOLL_CTL_ADD: errno=%d\n", errno); return; } nsos_adapt_poll_time = nsi_hws_get_time() + 1; nsi_hws_find_next_event(); } void nsos_adapt_poll_remove(struct nsos_mid_pollfd *pollfd) { int err; err = epoll_ctl(nsos_epoll_fd, EPOLL_CTL_DEL, pollfd->fd, NULL); if (err) { nsi_print_error_and_exit("error in EPOLL_CTL_DEL: errno=%d\n", errno); return; } nsos_adapt_nfds--; } void nsos_adapt_poll_update(struct nsos_mid_pollfd *pollfd) { struct pollfd fds = { .fd = pollfd->fd, .events = pollfd->events, }; int ret; ret = poll(&fds, 1, 0); if (ret < 0) { nsi_print_error_and_exit("error in poll(): errno=%d\n", errno); return; } if (ret > 0) { pollfd->revents = fds.revents; } } struct nsos_addrinfo_wrap { struct nsos_mid_addrinfo addrinfo_mid; struct nsos_mid_sockaddr_storage addr_storage; struct addrinfo *addrinfo; }; static int addrinfo_to_nsos_mid(struct addrinfo *res, struct nsos_mid_addrinfo **mid_res) { struct nsos_addrinfo_wrap *nsos_res_wraps; size_t idx_res = 0; size_t n_res = 0; int ret; for (struct addrinfo *res_p = res; res_p; res_p = res_p->ai_next) { n_res++; } if (n_res == 0) { return 0; } nsos_res_wraps = calloc(n_res, sizeof(*nsos_res_wraps)); if (!nsos_res_wraps) { return -NSOS_MID_ENOMEM; } for (struct addrinfo *res_p = res; res_p; res_p = res_p->ai_next, idx_res++) { struct nsos_addrinfo_wrap *wrap = &nsos_res_wraps[idx_res]; wrap->addrinfo = res_p; wrap->addrinfo_mid.ai_flags = res_p->ai_flags; ret = socket_family_to_nsos_mid(res_p->ai_family, &wrap->addrinfo_mid.ai_family); if (ret < 0) { goto free_wraps; } ret = socket_type_to_nsos_mid(res_p->ai_socktype, &wrap->addrinfo_mid.ai_socktype); if (ret < 0) { goto free_wraps; } ret = socket_proto_to_nsos_mid(res_p->ai_protocol, &wrap->addrinfo_mid.ai_protocol); if (ret < 0) { goto free_wraps; } wrap->addrinfo_mid.ai_addr = (struct nsos_mid_sockaddr *)&wrap->addr_storage; wrap->addrinfo_mid.ai_addrlen = sizeof(wrap->addr_storage); ret = sockaddr_to_nsos_mid(res_p->ai_addr, res_p->ai_addrlen, wrap->addrinfo_mid.ai_addr, &wrap->addrinfo_mid.ai_addrlen); if (ret < 0) { goto free_wraps; } wrap->addrinfo_mid.ai_canonname = res_p->ai_canonname ? strdup(res_p->ai_canonname) : NULL; wrap->addrinfo_mid.ai_next = &wrap[1].addrinfo_mid; } nsos_res_wraps[n_res - 1].addrinfo_mid.ai_next = NULL; *mid_res = &nsos_res_wraps->addrinfo_mid; return 0; free_wraps: for (struct nsos_mid_addrinfo *res_p = &nsos_res_wraps[0].addrinfo_mid; res_p; res_p = res_p->ai_next) { free(res_p->ai_canonname); } free(nsos_res_wraps); return ret; } int nsos_adapt_getaddrinfo(const char *node, const char *service, const struct nsos_mid_addrinfo *hints_mid, struct nsos_mid_addrinfo **res_mid, int *system_errno) { struct addrinfo hints; struct addrinfo *res = NULL; int ret; if (hints_mid) { hints.ai_flags = hints_mid->ai_flags; ret = socket_family_from_nsos_mid(hints_mid->ai_family, &hints.ai_family); if (ret < 0) { *system_errno = -ret; return NSOS_MID_EAI_SYSTEM; } ret = socket_type_from_nsos_mid(hints_mid->ai_socktype, &hints.ai_socktype); if (ret < 0) { *system_errno = -ret; return NSOS_MID_EAI_SYSTEM; } ret = socket_proto_from_nsos_mid(hints_mid->ai_protocol, &hints.ai_protocol); if (ret < 0) { *system_errno = -ret; return NSOS_MID_EAI_SYSTEM; } } ret = getaddrinfo(node, service, hints_mid ? &hints : NULL, &res); if (ret < 0) { return ret; } ret = addrinfo_to_nsos_mid(res, res_mid); if (ret < 0) { *system_errno = -ret; return NSOS_MID_EAI_SYSTEM; } return ret; } void nsos_adapt_freeaddrinfo(struct nsos_mid_addrinfo *res_mid) { struct nsos_addrinfo_wrap *wrap = CONTAINER_OF(res_mid, struct nsos_addrinfo_wrap, addrinfo_mid); for (struct nsos_mid_addrinfo *res_p = res_mid; res_p; res_p = res_p->ai_next) { free(res_p->ai_canonname); } freeaddrinfo(wrap->addrinfo); free(wrap); } int nsos_adapt_fcntl_getfl(int fd) { int flags; flags = fcntl(fd, F_GETFL); return fl_to_nsos_mid(flags); } int nsos_adapt_fcntl_setfl(int fd, int flags) { int ret; ret = fcntl(fd, F_SETFL, fl_from_nsos_mid(flags)); if (ret < 0) { return -errno_to_nsos_mid(errno); } return 0; } int nsos_adapt_fionread(int fd, int *avail) { int ret; ret = ioctl(fd, FIONREAD, avail); if (ret < 0) { return -errno_to_nsos_mid(errno); } return 0; } int nsos_adapt_dup(int oldfd) { int ret; ret = dup(oldfd); if (ret < 0) { return -errno_to_nsos_mid(errno); } return ret; } static void nsos_adapt_init(void) { nsos_epoll_fd = epoll_create(1); if (nsos_epoll_fd < 0) { nsi_print_error_and_exit("error from epoll_create(): errno=%d\n", errno); return; } } NSI_TASK(nsos_adapt_init, HW_INIT, 500); static void nsos_adapt_poll_triggered(void) { static struct epoll_event events[1024]; int ret; if (nsos_adapt_nfds == 0) { nsos_adapt_poll_time = NSI_NEVER; return; } ret = epoll_wait(nsos_epoll_fd, events, ARRAY_SIZE(events), 0); if (ret < 0) { if (errno == EINTR) { nsi_print_warning("interrupted epoll_wait()\n"); nsos_adapt_poll_time = nsi_hws_get_time() + 1; return; } nsi_print_error_and_exit("error in nsos_adapt poll(): errno=%d\n", errno); nsos_adapt_poll_time = NSI_NEVER; return; } for (int i = 0; i < ret; i++) { struct nsos_mid_pollfd *pollfd = events[i].data.ptr; pollfd->revents = nsos_epoll_to_poll_events(events[i].events); } if (ret > 0) { hw_irq_ctrl_set_irq(NSOS_IRQ); nsos_adapt_poll_time = nsi_hws_get_time() + 1; } else { nsos_adapt_poll_time = nsi_hws_get_time() + NSOS_EPOLL_WAIT_INTERVAL; } } NSI_HW_EVENT(nsos_adapt_poll_time, nsos_adapt_poll_triggered, 500);