/* * Copyright (C) 2024 Grenoble INP - ESISAR. All rights reserved. * SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception */ #include "platform_api_extension.h" #include "platform_api_vmcore.h" #include #include #include #include #include #include "libc_errno.h" #include #include // Static functions static bool textual_addr_to_sockaddr(const char *textual, int port, struct sockaddr *out, socklen_t *out_len) { struct sockaddr_in *v4; #ifdef IPPROTO_IPV6 struct sockaddr_in *v6; #endif assert(textual); v4 = (struct sockaddr_in *)out; if (zsock_inet_pton(AF_INET, textual, &v4->sin_addr.s_addr) == 1) { v4->sin_family = AF_INET; v4->sin_port = htons(port); *out_len = sizeof(struct sockaddr_in); return true; } #ifdef IPPROTO_IPV6 v6 = (struct sockaddr_in *)out; if (zsock_inet_pton(AF_INET6, textual, &v6->sin6_addr.s6_addr) == 1) { v6->sin6_family = AF_INET6; v6->sin6_port = htons(port); *out_len = sizeof(struct sockaddr_in); return true; } #endif return false; } static int sockaddr_to_bh_sockaddr(const struct sockaddr *sockaddr, bh_sockaddr_t *bh_sockaddr) { switch (sockaddr->sa_family) { case AF_INET: { struct sockaddr_in *addr = (struct sockaddr_in *)sockaddr; bh_sockaddr->port = ntohs(addr->sin_port); bh_sockaddr->addr_buffer.ipv4 = ntohl(addr->sin_addr.s_addr); bh_sockaddr->is_ipv4 = true; return BHT_OK; } #ifdef IPPROTO_IPV6 case AF_INET6: { struct sockaddr_in *addr = (struct sockaddr_in *)sockaddr; size_t i; bh_sockaddr->port = ntohs(addr->sin6_port); for (i = 0; i < sizeof(bh_sockaddr->addr_buffer.ipv6) / sizeof(bh_sockaddr->addr_buffer.ipv6[0]); i++) { uint16 part_addr = addr->sin6_addr.s6_addr[i * 2] | (addr->sin6_addr.s6_addr[i * 2 + 1] << 8); bh_sockaddr->addr_buffer.ipv6[i] = ntohs(part_addr); } bh_sockaddr->is_ipv4 = false; return BHT_OK; } #endif default: errno = EAFNOSUPPORT; return BHT_ERROR; } } static void bh_sockaddr_to_sockaddr(const bh_sockaddr_t *bh_sockaddr, struct sockaddr_storage *sockaddr, socklen_t *socklen) { if (bh_sockaddr->is_ipv4) { struct sockaddr_in *addr = (struct sockaddr_in *)sockaddr; addr->sin_port = htons(bh_sockaddr->port); addr->sin_family = AF_INET; addr->sin_addr.s_addr = htonl(bh_sockaddr->addr_buffer.ipv4); *socklen = sizeof(*addr); } #ifdef IPPROTO_IPV6 else { struct sockaddr_in6 *addr = (struct sockaddr_in6 *)sockaddr; size_t i; addr->sin6_port = htons(bh_sockaddr->port); addr->sin6_family = AF_INET6; for (i = 0; i < sizeof(bh_sockaddr->addr_buffer.ipv6) / sizeof(bh_sockaddr->addr_buffer.ipv6[0]); i++) { uint16 part_addr = htons(bh_sockaddr->addr_buffer.ipv6[i]); addr->sin6_addr.s6_addr[i * 2] = 0xff & part_addr; addr->sin6_addr.s6_addr[i * 2 + 1] = (0xff00 & part_addr) >> 8; } *socklen = sizeof(*addr); } #endif } static int getaddrinfo_error_to_errno(int error) { switch (error) { case DNS_EAI_AGAIN: return EAGAIN; case DNS_EAI_FAIL: return EFAULT; case DNS_EAI_MEMORY: return ENOMEM; case DNS_EAI_SYSTEM: return errno; default: return EINVAL; } } static int is_addrinfo_supported(struct zsock_addrinfo *info) { return // Allow only IPv4 and IPv6 (info->ai_family == AF_INET || info->ai_family == AF_INET6) // Allow only UDP and TCP && (info->ai_socktype == SOCK_DGRAM || info->ai_socktype == SOCK_STREAM) && (info->ai_protocol == IPPROTO_TCP || info->ai_protocol == IPPROTO_UDP); } static int os_socket_setbooloption(bh_socket_t socket, int level, int optname, bool is_enabled) { int option = (int)is_enabled; if (zsock_setsockopt(socket->fd, level, optname, &option, sizeof(option)) != 0) { return BHT_ERROR; } return BHT_OK; } static int os_socket_getbooloption(bh_socket_t socket, int level, int optname, bool *is_enabled) { assert(is_enabled); int optval; socklen_t optval_size = sizeof(optval); if (zsock_getsockopt(socket->fd, level, optname, &optval, &optval_size) != 0) { return BHT_ERROR; } *is_enabled = (bool)optval; return BHT_OK; } // Platform API implementation int os_socket_create(bh_socket_t *sock, bool is_ipv4, bool is_tcp) { int af = is_ipv4 ? AF_INET : AF_INET6; // now socket is a struct try *(sock)->fd *(sock) = BH_MALLOC(sizeof(zephyr_handle)); if (!*sock) { return BHT_ERROR; } if (is_tcp) { (*sock)->fd = zsock_socket(af, SOCK_STREAM, IPPROTO_TCP); } else { (*sock)->fd = zsock_socket(af, SOCK_DGRAM, IPPROTO_UDP); // IPPROTO_UDP or 0 } (*sock)->is_sock = true; if ((*sock)->fd == -1) { BH_FREE(*sock); return BHT_ERROR; } return BHT_OK; } int os_socket_bind(bh_socket_t socket, const char *host, int *port) { struct sockaddr_storage addr = { 0 }; socklen_t socklen; int ret; assert(host); assert(port); if (!textual_addr_to_sockaddr(host, *port, (struct sockaddr *)&addr, &socklen)) { return BHT_ERROR; } // F_SETF_SETFD and FD_CLOEXEC are not defined in zephyr. // SO_LINGER: Socket lingers on close (ignored, for compatibility) ret = zsock_bind(socket->fd, (struct sockaddr *)&addr, socklen); if (ret < 0) { return BHT_ERROR; } socklen = sizeof(addr); if (zsock_getsockname(socket->fd, (void *)&addr, &socklen) == -1) { return BHT_ERROR; } if (addr.ss_family == AF_INET) { // addr.sin_family *port = ntohs(((struct sockaddr_in *)&addr)->sin_port); } else { #ifdef IPPROTO_IPV6 *port = ntohs(((struct sockaddr_in *)&addr)->sin6_port); #else return BHT_ERROR; #endif } return BHT_OK; } int os_socket_settimeout(bh_socket_t socket, uint64 timeout_us) { struct timeval timeout = { 0 }; timeout.tv_sec = timeout_us / 1000000; timeout.tv_usec = timeout_us % 1000000; if (zsock_setsockopt(socket->fd, SOL_SOCKET, SO_RCVTIMEO, &timeout, sizeof(timeout)) != 0) { return BHT_ERROR; } return BHT_OK; } int os_socket_listen(bh_socket_t socket, int max_client) { return zsock_listen(socket->fd, max_client) != 0 ? BHT_ERROR : BHT_OK; } int os_socket_accept(bh_socket_t server_sock, bh_socket_t *sock, void *addr, unsigned int *addrlen) { *sock = BH_MALLOC(sizeof(zephyr_handle)); if (!*sock) { return BHT_ERROR; } (*sock)->fd = zsock_accept(server_sock->fd, addr, addrlen); if ((*sock)->fd < 0) { BH_FREE(*sock); return BHT_ERROR; } (*sock)->is_sock = true; return BHT_OK; } int os_socket_connect(bh_socket_t socket, const char *addr, int port) { struct sockaddr_storage addr_in = { 0 }; socklen_t socklen; int ret; assert(addr); if (!textual_addr_to_sockaddr(addr, port, (struct sockaddr *)&addr_in, &socklen)) { return BHT_ERROR; } ret = zsock_connect(socket->fd, (struct sockaddr *)&addr_in, socklen); if (ret < 0) { return BHT_ERROR; } return BHT_OK; } int os_socket_recv(bh_socket_t socket, void *buf, unsigned int len) { return zsock_recv(socket->fd, buf, len, 0); } int os_socket_recv_from(bh_socket_t socket, void *buf, unsigned int len, int flags, bh_sockaddr_t *src_addr) { struct sockaddr_storage sock_addr = { 0 }; socklen_t socklen = sizeof(sock_addr); int ret; ret = zsock_recvfrom(socket->fd, buf, len, flags, (struct sockaddr *)&sock_addr, &socklen); if (ret < 0) { return BHT_ERROR; } if (src_addr && socklen > 0) { // zsock_recvfrom doesn't seem to set `addr->sa_family`, // so we set it manually. ((struct sockaddr *)&sock_addr)->sa_family = src_addr->is_ipv4 == true ? AF_INET : AF_INET6; if (sockaddr_to_bh_sockaddr((struct sockaddr *)&sock_addr, src_addr) == BHT_ERROR) { return -1; } } else if (src_addr) { memset(src_addr, 0, sizeof(*src_addr)); } return ret; } int os_socket_send(bh_socket_t socket, const void *buf, unsigned int len) { return zsock_send(socket->fd, buf, len, 0); } int os_socket_send_to(bh_socket_t socket, const void *buf, unsigned int len, int flags, const bh_sockaddr_t *dest_addr) { struct sockaddr_storage addr = { 0 }; socklen_t socklen; (void)bh_sockaddr_to_sockaddr(dest_addr, &addr, &socklen); return zsock_sendto(socket->fd, buf, len, flags, (struct sockaddr *)&addr, socklen); } int os_socket_close(bh_socket_t socket) { zsock_close(socket->fd); BH_FREE(socket); return BHT_OK; } __wasi_errno_t os_socket_shutdown(bh_socket_t socket) { if (zsock_shutdown(socket->fd, ZSOCK_SHUT_RDWR) == -1) { return convert_errno(errno); } return __WASI_ESUCCESS; } int os_socket_inet_network(bool is_ipv4, const char *cp, bh_ip_addr_buffer_t *out) { if (!cp) return BHT_ERROR; if (is_ipv4) { if (zsock_inet_pton(AF_INET, cp, &out->ipv4) != 1) { return BHT_ERROR; } /* Note: ntohl(INADDR_NONE) == INADDR_NONE */ out->ipv4 = ntohl(out->ipv4); } else { #ifdef IPPROTO_IPV6 if (zsock_inet_pton(AF_INET6, cp, out->ipv6) != 1) { return BHT_ERROR; } for (int i = 0; i < 8; i++) { out->ipv6[i] = ntohs(out->ipv6[i]); } #else errno = EAFNOSUPPORT; return BHT_ERROR; #endif } return BHT_OK; } int os_socket_addr_resolve(const char *host, const char *service, uint8_t *hint_is_tcp, uint8_t *hint_is_ipv4, bh_addr_info_t *addr_info, size_t addr_info_size, size_t *max_info_size) { struct zsock_addrinfo hints = { 0 }, *res, *result; int hints_enabled = hint_is_tcp || hint_is_ipv4; int ret; size_t pos = 0; if (hints_enabled) { if (hint_is_ipv4) { hints.ai_family = *hint_is_ipv4 ? AF_INET : AF_INET6; } if (hint_is_tcp) { hints.ai_socktype = *hint_is_tcp ? SOCK_STREAM : SOCK_DGRAM; } } ret = zsock_getaddrinfo(host, strlen(service) == 0 ? NULL : service, hints_enabled ? &hints : NULL, &result); if (ret != BHT_OK) { errno = getaddrinfo_error_to_errno(ret); return BHT_ERROR; } res = result; while (res) { if (addr_info_size > pos) { if (!is_addrinfo_supported(res)) { res = res->ai_next; continue; } ret = sockaddr_to_bh_sockaddr(res->ai_addr, &addr_info[pos].sockaddr); if (ret == BHT_ERROR) { zsock_freeaddrinfo(result); return BHT_ERROR; } addr_info[pos].is_tcp = res->ai_socktype == SOCK_STREAM; } pos++; res = res->ai_next; } *max_info_size = pos; zsock_freeaddrinfo(result); return BHT_OK; } int os_socket_addr_local(bh_socket_t socket, bh_sockaddr_t *sockaddr) { struct sockaddr_storage addr_storage = { 0 }; socklen_t addr_len = sizeof(addr_storage); int ret; ret = zsock_getsockname(socket->fd, (struct sockaddr *)&addr_storage, &addr_len); if (ret != BHT_OK) { return BHT_ERROR; } return sockaddr_to_bh_sockaddr((struct sockaddr *)&addr_storage, sockaddr); } int os_socket_addr_remote(bh_socket_t socket, bh_sockaddr_t *sockaddr) { struct sockaddr_storage addr_storage = { 0 }; socklen_t addr_len = sizeof(addr_storage); int ret; ret = zsock_getpeername(socket->fd, (struct sockaddr *)&addr_storage, &addr_len); if (ret != BHT_OK) { return BHT_ERROR; } return sockaddr_to_bh_sockaddr((struct sockaddr *)&addr_storage, sockaddr); } int os_socket_set_send_buf_size(bh_socket_t socket, size_t bufsiz) { int buf_size_int = (int)bufsiz; if (zsock_setsockopt(socket->fd, SOL_SOCKET, SO_SNDBUF, &buf_size_int, sizeof(buf_size_int)) != 0) { return BHT_ERROR; } return BHT_OK; } int os_socket_get_send_buf_size(bh_socket_t socket, size_t *bufsiz) { assert(bufsiz); int buf_size_int; socklen_t bufsiz_len = sizeof(buf_size_int); if (zsock_getsockopt(socket->fd, SOL_SOCKET, SO_SNDBUF, &buf_size_int, &bufsiz_len) != 0) { return BHT_ERROR; } *bufsiz = (size_t)buf_size_int; return BHT_OK; } int os_socket_set_recv_buf_size(bh_socket_t socket, size_t bufsiz) { if (zsock_setsockopt(socket->fd, SOL_SOCKET, SO_RCVBUF, &bufsiz, sizeof(bufsiz)) != 0) { return BHT_ERROR; } return BHT_OK; } int os_socket_get_recv_buf_size(bh_socket_t socket, size_t *bufsiz) { assert(bufsiz); int buf_size_int; socklen_t bufsiz_len = sizeof(buf_size_int); if (zsock_getsockopt(socket->fd, SOL_SOCKET, SO_RCVBUF, &buf_size_int, &bufsiz_len) != 0) { return BHT_ERROR; } *bufsiz = (size_t)buf_size_int; return BHT_OK; } int os_socket_set_keep_alive(bh_socket_t socket, bool is_enabled) { return os_socket_setbooloption(socket, SOL_SOCKET, SO_KEEPALIVE, is_enabled); } int os_socket_get_keep_alive(bh_socket_t socket, bool *is_enabled) { return os_socket_getbooloption(socket, SOL_SOCKET, SO_KEEPALIVE, is_enabled); } int os_socket_set_send_timeout(bh_socket_t socket, uint64 timeout_us) { struct timeval tv; tv.tv_sec = timeout_us / 1000000UL; tv.tv_usec = timeout_us % 1000000UL; if (zsock_setsockopt(socket->fd, SOL_SOCKET, SO_SNDTIMEO, &tv, sizeof(tv)) != 0) { return BHT_ERROR; } return BHT_OK; } int os_socket_get_send_timeout(bh_socket_t socket, uint64 *timeout_us) { struct timeval tv; socklen_t tv_len = sizeof(tv); if (zsock_getsockopt(socket->fd, SOL_SOCKET, SO_SNDTIMEO, &tv, &tv_len) != 0) { return BHT_ERROR; } *timeout_us = (tv.tv_sec * 1000000UL) + tv.tv_usec; return BHT_OK; } int os_socket_set_recv_timeout(bh_socket_t socket, uint64 timeout_us) { struct timeval tv; tv.tv_sec = timeout_us / 1000000UL; tv.tv_usec = timeout_us % 1000000UL; if (zsock_setsockopt(socket->fd, SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(tv)) != 0) { return BHT_ERROR; } return BHT_OK; } int os_socket_get_recv_timeout(bh_socket_t socket, uint64 *timeout_us) { struct timeval tv; socklen_t tv_len = sizeof(tv); if (zsock_getsockopt(socket->fd, SOL_SOCKET, SO_RCVTIMEO, &tv, &tv_len) != 0) { return BHT_ERROR; } *timeout_us = (tv.tv_sec * 1000000UL) + tv.tv_usec; return BHT_OK; } int os_socket_set_reuse_addr(bh_socket_t socket, bool is_enabled) { return os_socket_setbooloption(socket, SOL_SOCKET, SO_REUSEADDR, is_enabled); } int os_socket_get_reuse_addr(bh_socket_t socket, bool *is_enabled) { return os_socket_getbooloption(socket, SOL_SOCKET, SO_REUSEADDR, is_enabled); } int os_socket_set_reuse_port(bh_socket_t socket, bool is_enabled) { return os_socket_setbooloption(socket, SOL_SOCKET, SO_REUSEPORT, is_enabled); } int os_socket_get_reuse_port(bh_socket_t socket, bool *is_enabled) { return os_socket_getbooloption(socket, SOL_SOCKET, SO_REUSEPORT, is_enabled); } // SO_LINGER Socket lingers on close (ignored, for compatibility) int os_socket_set_linger(bh_socket_t socket, bool is_enabled, int linger_s) { errno = ENOSYS; return BHT_ERROR; } int os_socket_get_linger(bh_socket_t socket, bool *is_enabled, int *linger_s) { errno = ENOSYS; return BHT_ERROR; } int os_socket_set_tcp_no_delay(bh_socket_t socket, bool is_enabled) { return os_socket_setbooloption(socket, IPPROTO_TCP, TCP_NODELAY, is_enabled); } int os_socket_get_tcp_no_delay(bh_socket_t socket, bool *is_enabled) { return os_socket_getbooloption(socket, IPPROTO_TCP, TCP_NODELAY, is_enabled); } int os_socket_set_tcp_quick_ack(bh_socket_t socket, bool is_enabled) { errno = ENOSYS; return BHT_ERROR; } int os_socket_get_tcp_quick_ack(bh_socket_t socket, bool *is_enabled) { errno = ENOSYS; return BHT_ERROR; } int os_socket_set_tcp_keep_idle(bh_socket_t socket, uint32_t time_s) { int time_s_int = (int)time_s; #ifdef TCP_KEEPIDLE if (zsock_setsockopt(socket->fd, IPPROTO_TCP, TCP_KEEPIDLE, &time_s_int, sizeof(time_s_int)) != 0) { return BHT_ERROR; } return BHT_OK; #elif defined(TCP_KEEPALIVE) if (zsock_setsockopt(socket->fd, IPPROTO_TCP, TCP_KEEPALIVE, &time_s_int, sizeof(time_s_int)) != 0) { return BHT_ERROR; } return BHT_OK; #else errno = ENOSYS; return BHT_ERROR; #endif } int os_socket_get_tcp_keep_idle(bh_socket_t socket, uint32_t *time_s) { assert(time_s); int time_s_int; socklen_t time_s_len = sizeof(time_s_int); #ifdef TCP_KEEPIDLE if (zsock_getsockopt(socket->fd, IPPROTO_TCP, TCP_KEEPIDLE, &time_s_int, &time_s_len) != 0) { return BHT_ERROR; } *time_s = (uint32)time_s_int; return BHT_OK; #elif defined(TCP_KEEPALIVE) if (zsock_getsockopt(socket->fd, IPPROTO_TCP, TCP_KEEPALIVE, &time_s_int, &time_s_len) != 0) { return BHT_ERROR; } *time_s = (uint32)time_s_int; return BHT_OK; #else errno = ENOSYS; return BHT_ERROR; #endif } int os_socket_set_tcp_keep_intvl(bh_socket_t socket, uint32_t time_s) { int time_s_int = (int)time_s; #ifdef TCP_KEEPINTVL if (zsock_setsockopt(socket->fd, IPPROTO_TCP, TCP_KEEPINTVL, &time_s_int, sizeof(time_s_int)) != 0) { return BHT_ERROR; } return BHT_OK; #else errno = ENOSYS; return BHT_ERROR; #endif } int os_socket_get_tcp_keep_intvl(bh_socket_t socket, uint32_t *time_s) { #ifdef TCP_KEEPINTVL if (!socket || !time_s || socket->fd < 0) { errno = EINVAL; return BHT_ERROR; } int time_s_int; socklen_t time_s_len = sizeof(time_s_int); if (zsock_getsockopt(socket->fd, IPPROTO_TCP, TCP_KEEPINTVL, &time_s_int, &time_s_len) != 0) { return BHT_ERROR; } if (time_s_int < 0) { errno = EINVAL; return BHT_ERROR; } *time_s = (uint32_t)time_s_int; return BHT_OK; #else errno = ENOSYS; return BHT_ERROR; #endif } int os_socket_set_tcp_fastopen_connect(bh_socket_t socket, bool is_enabled) { errno = ENOSYS; return BHT_ERROR; } int os_socket_get_tcp_fastopen_connect(bh_socket_t socket, bool *is_enabled) { errno = ENOSYS; return BHT_ERROR; } int os_socket_set_ip_multicast_loop(bh_socket_t socket, bool ipv6, bool is_enabled) { if (ipv6) { #ifdef IPPROTO_IPV6 return os_socket_setbooloption(socket, IPPROTO_IPV6, IPV6_MULTICAST_LOOP, is_enabled); #else errno = EAFNOSUPPORT; return BHT_ERROR; #endif } else { return os_socket_setbooloption(socket, IPPROTO_IP, IP_MULTICAST_LOOP, is_enabled); } } int os_socket_get_ip_multicast_loop(bh_socket_t socket, bool ipv6, bool *is_enabled) { if (ipv6) { #ifdef IPPROTO_IPV6 return os_socket_getbooloption(socket, IPPROTO_IPV6, IPV6_MULTICAST_LOOP, is_enabled); #else errno = EAFNOSUPPORT; return BHT_ERROR; #endif } else { return os_socket_getbooloption(socket, IPPROTO_IP, IP_MULTICAST_LOOP, is_enabled); } } int os_socket_set_ip_add_membership(bh_socket_t socket, bh_ip_addr_buffer_t *imr_multiaddr, uint32_t imr_interface, bool is_ipv6) { assert(imr_multiaddr); if (is_ipv6) { #if defined(IPPROTO_IPV6) && !defined(BH_PLATFORM_COSMOPOLITAN) struct ipv6_mreq mreq; for (int i = 0; i < 8; i++) { ((uint16_t *)mreq.ipv6mr_multiaddr.s6_addr)[i] = imr_multiaddr->ipv6[i]; } mreq.ipv6mr_interface = imr_interface; if (setsockopt(socket->fd, IPPROTO_IPV6, IPV6_ADD_MEMBERSHIP, &mreq, sizeof(mreq)) != 0) { return BHT_ERROR; } #else errno = EAFNOSUPPORT; return BHT_ERROR; #endif } else { struct ip_mreqn mreq; mreq.imr_multiaddr.s_addr = imr_multiaddr->ipv4; mreq.imr_address.s_addr = imr_interface; if (zsock_setsockopt(socket->fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &mreq, sizeof(mreq)) != 0) { return BHT_ERROR; } } return BHT_OK; } int os_socket_set_ip_drop_membership(bh_socket_t socket, bh_ip_addr_buffer_t *imr_multiaddr, uint32_t imr_interface, bool is_ipv6) { assert(imr_multiaddr); if (is_ipv6) { #if defined(IPPROTO_IPV6) && !defined(BH_PLATFORM_COSMOPOLITAN) struct ipv6_mreq mreq; for (int i = 0; i < 8; i++) { ((uint16_t *)mreq.ipv6mr_multiaddr.s6_addr)[i] = imr_multiaddr->ipv6[i]; } mreq.ipv6mr_interface = imr_interface; if (zsock_setsockopt(socket->fd, IPPROTO_IPV6, IPV6_DROP_MEMBERSHIP, &mreq, sizeof(mreq)) != 0) { return BHT_ERROR; } #else errno = EAFNOSUPPORT; return BHT_ERROR; #endif } else { struct ip_mreqn mreq; mreq.imr_multiaddr.s_addr = imr_multiaddr->ipv4; mreq.imr_address.s_addr = imr_interface; if (zsock_setsockopt(socket->fd, IPPROTO_IP, IP_DROP_MEMBERSHIP, &mreq, sizeof(mreq)) != 0) { return BHT_ERROR; } } return BHT_OK; } int os_socket_set_ip_ttl(bh_socket_t socket, uint8_t ttl_s) { if (zsock_setsockopt(socket->fd, IPPROTO_IP, IP_TTL, &ttl_s, sizeof(ttl_s)) != 0) { return BHT_ERROR; } return BHT_OK; } int os_socket_get_ip_ttl(bh_socket_t socket, uint8_t *ttl_s) { /* Most socket-level options utilize an int argument for optval */ int opt; socklen_t opt_len = sizeof(opt); if (zsock_getsockopt(socket->fd, IPPROTO_IP, IP_TTL, &opt, &opt_len) != 0) { return BHT_ERROR; } *ttl_s = (uint8_t)opt; return BHT_OK; } int os_socket_set_ip_multicast_ttl(bh_socket_t socket, uint8_t ttl_s) { if (zsock_setsockopt(socket->fd, IPPROTO_IP, IP_MULTICAST_TTL, &ttl_s, sizeof(ttl_s)) != 0) { return BHT_ERROR; } return BHT_OK; } int os_socket_get_ip_multicast_ttl(bh_socket_t socket, uint8_t *ttl_s) { /* Most socket-level options utilize an int argument for optval */ int opt; socklen_t opt_len = sizeof(opt); if (zsock_getsockopt(socket->fd, IPPROTO_IP, IP_MULTICAST_TTL, &opt, &opt_len) != 0) { return BHT_ERROR; } *ttl_s = (uint8_t)opt; return BHT_OK; } int os_socket_set_ipv6_only(bh_socket_t socket, bool is_enabled) { #ifdef IPPROTO_IPV6 return os_socket_setbooloption(socket, IPPROTO_IPV6, IPV6_V6ONLY, is_enabled); #else errno = EAFNOSUPPORT; return BHT_ERROR; #endif } int os_socket_get_ipv6_only(bh_socket_t socket, bool *is_enabled) { #ifdef IPPROTO_IPV6 return os_socket_getbooloption(socket, IPPROTO_IPV6, IPV6_V6ONLY, is_enabled); #else errno = EAFNOSUPPORT; return BHT_ERROR; #endif } int os_socket_set_broadcast(bh_socket_t socket, bool is_enabled) { return os_socket_setbooloption(socket, SOL_SOCKET, SO_BROADCAST, is_enabled); } int os_socket_get_broadcast(bh_socket_t socket, bool *is_enabled) { return os_socket_getbooloption(socket, SOL_SOCKET, SO_BROADCAST, is_enabled); } // Experimental : int os_ioctl(os_file_handle handle, int request, ...) { return __WASI_ENOSYS; // int ret = -1; // va_list args; // va_start(args, request); // ret = zsock_ioctl(handle->fd, request, args); // va_end(args); // return ret; } int os_poll(os_poll_file_handle *fds, os_nfds_t nfs, int timeout) { return zsock_poll(fds, nfs, timeout); }